http://acm.hdu.edu.cn/showproblem.php?pid=3256
2009 宁波网赛题。。此次网赛题难度相当不一般啊。。
题意:给定500条线段,以浮点型输入最多4为小数。。。问这些线段覆盖的整数点有多少个。。。。数据保证在-10000<x, y<10000。。。
分析:需要判重,所以不能直接做。。。总可能点数达到4*10^8个,所以不可能全部标记。。。因为线段条数不多,范围也很小,可以直接枚举x的坐标标记y坐标被覆盖的个数。。。但如果标记每次都初始化为0的话复杂度4*10^8必然超时,这里有点小技巧就是flag标记值值初始化一次,然后更新为当前的x坐标值即可。。这样就是500*20000的复杂度了。。。
此题还比较变态啊。。。输入的数据由于double精度问题不能直接输入。。。所以只能变换输入方式使得所有值都是精确地整数值。。。
还有。。大于0和小于0时出现取余是一定要注意分别判断。。。wa了好久啊。。。
用double做死定过不了啊。。。郁闷。。。
代码:3000+ms
#include <string.h> #include <stdio.h> #include <algorithm> #include <cmath> #include <iostream> using namespace std; const int M=10010; const int N=510; char s[110]; struct Line { __int64 x1, y1, x2, y2; } ll[N], pos; int n, flag[M*2], ans; void input(__int64 &a) { scanf("%s", &s); int i, j, len=strlen(s); a = 0; i = 0; if(s[0]=='-') i++; for(; i<len && s[i]!='.'; i++) { a *= 10; a += s[i]-'0'; } for(i++, j=0; i<len; i++, j++) { a *= 10; a += s[i]-'0'; } while(j<4) { a *= 10; j++; } if(s[0]=='-') a = -a; } void cal(int i, Line li) { if(li.x1==li.x2) { if(li.x1==i*10000) { if(li.y1>li.y2) swap(li.y1, li.y2); if(li.y1<=0) //小于0的情况没考虑,这里wa了好久啊。。。 li.y1 = li.y1/10000; else li.y1 = li.y1/10000+1; if(li.y2>=0) //... li.y2 /= 10000; else li.y2 = li.y2/10000-1; for(int j=li.y1; j<=li.y2; j++) { if(flag[j+M]!=i) { flag[j+M] = i; ans++; } } } return; } if(!(li.x1<=i*10000 && i*10000<=li.x2) ) return; if((i*10000-li.x1)*(li.y2-li.y1)%(li.x2-li.x1)==0) { int j = (i*10000-li.x1)*(li.y2-li.y1)/(li.x2-li.x1)+li.y1; if(j%10000==0) { j /= 10000; if(flag[j+M]!=i) { flag[j+M] = i; ans++; } } } } int main() { int i, j, cas, cas1; scanf("%d", &cas); for(cas1=1; cas1<=cas; cas1++) { scanf("%d", &n); pos.x1 = M*M*10; pos.x2 = -M*M*10; for(i=0; i<n; i++) { input(ll[i].x1); input(ll[i].y1); input(ll[i].x2); input(ll[i].y2); if(ll[i].x1 > ll[i].x2) { swap(ll[i].x1, ll[i].x2); swap(ll[i].y1, ll[i].y2); } if(ll[i].x1<pos.x1) pos.x1 = ll[i].x1; if(ll[i].x2<pos.x1) pos.x1 = ll[i].x2; if(ll[i].x1>pos.x2) pos.x2 = ll[i].x1; if(ll[i].x2>pos.x2) pos.x2 = ll[i].x2; } for(i=0; i<M*2; i++) flag[i] = -1000000; ans = 0; pos.x1 /= 10000; pos.x1--; //小于0的时候。。。wa啊。。。 pos.x2 /= 10000; pos.x2++; for(i=pos.x1; i<=pos.x2; i++) { for(j=0; j<n; j++) { cal(i, ll[j]); } } printf("Case %d: %d\n", cas1, ans); } return 0; }
代码:double做的死wa。。
#include <string.h> #include <stdio.h> #include <algorithm> #include <cmath> #include <set> #include <iostream> using namespace std; const double eps=0.00005; int M=10001; const int N=510; struct Point { double x, y; }; struct Line { Point a, b; } line[N]; int n, ans, flag[20010]; double xn, xx; inline double ffabs(double x) { if(x<0) return -x; return x; } void cal(int i, Line li) { double ii=i; if(ffabs(li.a.x-li.b.x)<eps) { if(ffabs(li.a.x-ii)<eps) { int yn, yx; if(li.a.y>li.b.y) swap(li.a, li.b); if(li.a.y<=0) yn = li.a.y-eps; else yn = li.a.y-eps+1; if(li.b.y>=0) yx = li.b.y+eps; else yx = li.b.y+eps-1; for(int j=yn; j<=yx; j++) { if(flag[j+M]!=i) { ans++; flag[j+M] = i; } } } return; } double y, y1; int yn; if(li.a.x-ii>eps || ii-li.b.x>eps) return; y = (i-li.a.x)*(li.b.y-li.a.y)/(li.b.x-li.a.x)+li.a.y; yn = y; if(ffabs(y-yn)<0.00000001) { if(flag[yn+M]!=i) { flag[yn+M] = i; ans++; } } } int main() { int i, j, cas, cas1; scanf("%d", &cas); for(cas1=1; cas1<=cas; cas1++) { scanf("%d", &n); xx = -10010; xn = 10010; for(i=0; i<n; i++) { scanf("%lf%lf%lf%lf", &line[i].a.x, &line[i].a.y, &line[i].b.x, &line[i].b.y); if(line[i].a.x>line[i].b.x) swap(line[i].a, line[i].b); //x从小到大。。。 if(line[i].b.x>xx) xx = line[i].b.x; if(line[i].a.x<xn) xn = line[i].a.x; } ans = 0; for(i=0; i<20010; i++) flag[i] = -1000000; for(i=xn-1; i<=xx+1; i++) { for(j=0; j<n; j++) cal(i, line[j]); } printf("Case %d: %d\n", cas1, ans); } return 0; }