随笔-21  评论-10  文章-21  trackbacks-0
pku 3301 texas trip 最小正方形覆盖
最小矩形覆盖有个良好的性质,就是最小矩形必定至少一条边上有两个点,这样就可以很方便的枚举,
但我没有看到最小正方形的类似的性质
这样的话连枚举的算法都很困难
标程的算法是枚举转角,先将[0, PI/2]等分1000份,选个最大的,再将这附近的角度等分1000份,其他的就舍弃了,然后迭代下去,这样做应该是有风险的,但cas最多只有30个点,等分的单位角度够小的话,正确率还是很高的
关键是没有更好的枚举方法,我们只能枚举连续的转角,但实际上我们不能枚举连续的东西,只能迭代逼近,这里面也可能发生错误

 1 waterloo标程
 2 #include <stdio.h>
 3 #include <math.h>
 4 #include <assert.h>
 5 
 6 #define N 1000
 7 
 8 double x[100], y[100], xx, yy, base,scale;
 9 int rep,n,i,j,k,T;
10 
11 main(){
12    double minx, maxx, miny, maxy, dx, dy, dd, best; int besti;
13    scanf("%d\n",&T);
14    while (T--) {
15       assert (1 == scanf("%d",&n));
16       for (i=0;i<n;i++) scanf("%lf%lf",&x[i],&y[i]);
17       best = 1000;
18       base = 0; scale=1;
19       for (rep=0;rep<10;rep++) {
20          besti = 0;
21          for (i=0;i<N;i++) {
22             double sininc = sin(base+scale*i*M_PI/2/N);
23             double cosinc = cos(base+scale*i*M_PI/2/N);
24             maxx = maxy = -1e99; minx = miny = 1e99;
25             for (j=0;j<n;j++){
26                xx = cosinc*x[j] - sininc*y[j];
27                yy = sininc*x[j] + cosinc*y[j];
28                //printf("x y xx yy %lg %lg %lg %lg\n",x[j],y[j],xx,yy);
29                if (xx < minx) minx = xx;
30                if (xx > maxx) maxx = xx;
31                if (yy < miny) miny = yy;
32                if (yy > maxy) maxy = yy;
33             }
34             dx = maxx - minx;
35             dy = maxy - miny;
36             dd = dx;
37             if (dy > dd) dd = dy;
38             if (dd < best) {
39                best = dd;
40                besti = i;
41             }
42             //printf("i %d dd %0.5lf\n",i,dd);
43          }
44          base += scale*(besti-1)*M_PI/2/N;
45          scale = scale/N*2;
46       }
47       printf("%0.2lf\n",best*best);
48    }
49    assert(1 != scanf(" %c",&i));
50 }
51 

posted on 2009-02-28 16:56 wangzhihao 阅读(526) 评论(0)  编辑 收藏 引用

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理