題目來自nyist第16題,如下:
描述
有n個矩形,每個矩形可以用a,b來描述,表示長和寬。矩形X(a,b)可以嵌套在矩形Y(c,d)中當且僅當a<c,b<d或者b<c,a<d(相當于旋轉X90度)。例如(1,5)可以嵌套在(6,2)內,但不能嵌套在(3,4)中。你的任務是選出盡可能多的矩形排成一行,使得除最后一個外,每一個矩形都可以嵌套在下一個矩形內。輸入第一行是一個正正數N(0<N<10),表示測試數據組數,每組測試數據的第一行是一個正正數n,表示該組測試數據中含有矩形的個數(n<=1000)隨后的n行,每行有兩個數a,b(0<a,b<100),表示矩形的長和寬輸出每組測試數據都輸出一個數,表示最多符合條件的矩形數目,每組輸出占一行
要能實現嵌套,首先要將所有矩形按照大小排序,并先將矩形旋轉成長寬方向一致。排序使用sort函數,由于排序之后只可能前面的矩形嵌套在后面的矩形之中,故用DP解法遞推,雙循環i,j(循環結束標志i<n,j<i)即對第i個矩形,循環查找其前面的i-1個矩形,找出能被其嵌套并且對應dp數組的值+1后大于dp[i]的,dp數組在本題中指嵌套在第i個矩形中的矩形個數(我在程序中初始化為1)。找出后更新dp[i]的值。計算完dp數組之后,找出其中的最大值,即為嵌套矩形的最大層數。
代碼如下:
#include<cstdio>#include<algorithm>using namespace std;struct Re{ int a,b;}rectangle[1000+5];int dp[1000+5];int cmp(struct Re x,struct Re y){ if(x.a == y.a) return x.b <= y.b; return x.a <= y.a;}int main(){ int n,N,i,j,k,t; scanf("%d",&N); while(N--) { scanf("%d",&n); for(i = 0;i < n;i++) { scanf("%d %d",&(rectangle[i].a),&(rectangle[i].b)); if(rectangle[i].a > rectangle[i].b) { t = rectangle[i].a; rectangle[i].a = rectangle[i].b; rectangle[i].b = t; } } sort(rectangle,rectangle+n,cmp); for(i = 0;i < n;i++) { dp[i] = 1; for(j = 0;j < i;j++) { if(rectangle[i].a > rectangle[j].a && rectangle[i].b > rectangle[j].b && (dp[j]+1) > dp[i]) dp[i] = dp[j]+1; } } int max = 0; //因為dp[i]初始化為0 for(i = 0;i < n;i++) max = max > dp[i] ? max : dp[i]; PRintf("%d/n",max); } return 0;} 本題還可以用貪心求解,方法是在排序之后,貪心選擇能包含最多的序列,雖然能跑過,但是我不會證明其正確性,有些頭疼。類似問題還有poj1065,可以作為練習。本題解法很常見,會經常用到。
新聞熱點
疑難解答