POJ 1131 Doing Windows
题意:
给定四个可以等比例缩放(缩放后长宽比不变)的矩形, 求在经过缩放后是否可以完全覆盖另一个矩形。
分析:知道要深搜,不敢做。。
看了解题报告的两幅图后豁然开朗,分两种,还是没用深搜;
一种是填充式,另一种是分隔式
1:填充式,每次放矩形时,放大至最大;
+------------+
|111111111111|
|222233333333|
|222233333333|
|222233333333|
|222244444444|
|222244444444|
+------------+
2:分隔式,2*2,每个小矩形至少和另一个有有一条公共边
+---+--------+
| | |
+---+---+----|
| | |
| | |
| | |
| | |
+-------+----+
结:我一直不相信浮点转整数的稳定性,因为不了解,怕出错,其实只要加上修正就可以;
如 int a = da+0.00005;fabs(a-da)<0.0001(具体视精度要求而定)
View Code
#include <cstdio> #include <algorithm> #include <cmath> using namespace std; double r[4]; int is_integer(double t) { int i=t+0.0000005; if(fabs(t-i)<0.000001) return i; return 0; } int try_height(int i,int sw) { double dh=sw/r[i]; int h=dh+.0000005; if(fabs(h-dh)<0.000001) return h; return -1; } int try_weight(int i,int sh) { double dw=sh*r[i]; int w=dw+.0000005; if(fabs(w-dw)<0.000001) return w; return -1; } bool fill(int sw, int sh) { int w,h,tw,th,j,k,a[4]={0,1,2,3}; for(j=0;j<24;j++) { tw=sw,th=sh; for(k=0;k<4&&th&&tw;k++) { h=try_height(a[k],tw); if(h!=-1&&h<=th) th-=h; else { w=try_weight(a[k],th); if(w==-1||w>tw) break; tw-=w; } } if (k==4&&(!tw||!th)) return true; next_permutation(a,a+4); } return false; } int fit(int w ,int i,int j) { double h=w/(r[i]+r[j]); int ret=is_integer(h); if(is_integer(h*r[i]) && is_integer(h*r[j])) return ret; return 0; } bool mixture(int sw,int sh) { int a[4]={0,1,2,3}; int i,j,k,h1,h2,tw,th; for(i=0;i<24;i++) { tw=sw,th=sh; h1=fit(tw,a[0],a[1]); h2=fit(tw,a[2],a[3]); if(h1&&h2&&h1+h2==sh) return true; next_permutation(a,a+4); } return false; } int main() { int i,t,w,h,sw,sh; for(t=1;scanf("%d%d",&sw,&sh),sw+sh;t++) { bool found=false; for(i=0;i<4;i++) { scanf("%d%d",&w,&h); r[i]=w*1.0/h; } if(fill(sw,sh)) found=true; else { if(mixture(sw,sh)) found=true; else { for(i=0;i<4;i++) r[i]=1/r[i]; found=mixture(sh,sw); } } if(found) printf("Set %d: Yes\n",t); else printf("Set %d: No\n",t); } return 0; }