問題
在一塊電路板的上下兩端分別有n個(gè)接線柱。根據(jù)電路設(shè)計(jì),要求用導(dǎo)線 (i,π(i)),將上端接線柱 i 與下端接線柱 π(i) 相連,
如圖,其中 π(i),1《=i《=n,是(1,2……,n)的一個(gè)排列。導(dǎo)線(i,π(i))稱為該電路板上的第i條連線。對(duì)于任何 1《=i《s《=n,第i條連線和第s條連線相交的充分且必要條件是 π(i) 》 π(s)。
ps:注意 π(pi) 和 n,不是小寫的n,別看錯(cuò)了
問:在制作電路板時(shí),要求將這n條線分布到若干個(gè)絕緣層上,在同一層上的連線不能相交。電路布線問題要確定將哪些連線安排在第一層上,使得該層上有盡可能多的連線。
詳細(xì)說明
首先 上下各有 n 個(gè)接線柱,用 a[i] 數(shù)組表示 與 上接線柱 相連線的 下接線柱,再用 set[i][j] 表示上接線柱 i 與下接線柱 j 相連線的 左邊(包括 i ,j 連線)的最大不相交連線的個(gè)數(shù)。
ps(這里的 j ,i 和上面問題中說的不是一個(gè)東西,這里的 i指的是上接線柱,j指的是下接線柱)
1、公式如下:
如果 j != a[i] 則 max( set[i-1][j],set[i][j-1] )
set[i,j] =
如果 j == a[i] 則 set[i-1][j-1] + 1
循環(huán)遍歷 i 和 j ,具體使用 i X j 的嵌套循環(huán),其實(shí)就是每一個(gè)上接線柱和每一個(gè)下接線柱做一次匹配,這樣就可以得出結(jié)果,set[n][n]即我們想要的結(jié)果。最后通過回溯把結(jié)果輸出出來
==》 j == a[i],表示當(dāng)上圖中的某個(gè)上下接線柱相連時(shí),這時(shí)這兩個(gè)點(diǎn)不會(huì)在和其他的接線柱相連,
這時(shí)在 (i ,j) 處的最大不相交連線的個(gè)數(shù)就應(yīng)該是就是 第 i-1個(gè)上接線柱 和第 j-1個(gè)下接線柱時(shí)的 最大不相交連線的個(gè)數(shù) + 1。這個(gè)+1很重要。
==》 j != a[i],表示當(dāng)上圖中的某個(gè)上下接線柱相連時(shí),這時(shí) i 點(diǎn)和 非 j 點(diǎn)相連, j 點(diǎn)和 非 i 點(diǎn)相連,
這時(shí)在( i ,j) 處的最大不相交連線的個(gè)數(shù)就應(yīng)該和 ( i - 1,j ) 處最大不相交連線的個(gè)數(shù)、 ( i,j - 1)處最大不相交連線的個(gè)數(shù)中較大的最大不相交連線的個(gè)數(shù)相同。
2、參考圖如下:
紅色標(biāo)明的就是算法選擇的路徑(向上優(yōu)先,也可以用向左優(yōu)先,答案都是四個(gè),但值會(huì)有一點(diǎn)不同)。在斜角值改變時(shí)可以取得所求的子集。即 9-》10,7-》9, 5-》5, 3-》4
代碼塊
#include 《stdio.h》
#include 《stdlib.h》
#define MAX( a, b ) ( (a) 》 (b) ? (a) : (b) )
void circut( int a[], int set[][11], int n );
void back_track( int i, int j, int set[][11] );
int main()
{
int a[] = { 0, 8, 7, 4, 2, 5, 1, 9, 3, 10, 6 };
int set[11][11];
circut( a, set, 10 );
printf( “max set: %d \n”, set[10][10] );
back_track( 10, 10, set );
printf( “\n” );
system( “pause” );
return(0);
}
void circut( int a[], int set[][11], int n )
{
int i, j;
for ( i = 0; i 《 n; i++ )
{
set[i][0] = 0;
set[0][i] = 0;
}
for ( i = 1; i 《= n; i++ )
{
for ( j = 1; j 《= n; j++ )
{
if ( a[i] != j )
set[i][j] = MAX( set[i - 1][j], set[i][j - 1] );
else
set[i][j] = set[i - 1][j - 1] + 1;
}
}
}
void back_track( int i, int j, int set[][11] )
{
if ( i == 0 )
return;
if ( set[i][j] == set[i - 1][j] )
back_track( i - 1, j, set );
else if ( set[i][j] == set[i][j - 1] )
back_track( i, j - 1, set );
else{
back_track( i - 1, j - 1, set );
printf( “(%d,%d) ”, i, j );
}
}
時(shí)間復(fù)雜度
circut 的時(shí)間復(fù)雜度為O(n2)
back_track的時(shí)間復(fù)雜度為 O(n)
如有錯(cuò)誤請(qǐng)指正。
-
電路板
+關(guān)注
關(guān)注
140文章
4980瀏覽量
98392 -
電路設(shè)計(jì)
+關(guān)注
關(guān)注
6678文章
2459瀏覽量
204970
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論