记录编号 158144 评测结果 AAAAAAAAAA
题目名称 [USACO Feb12] 对称 最终得分 100
用户昵称 Gravatarcstdio 是否通过 通过
代码语言 C++ 运行时间 0.008 s
提交时间 2015-04-12 21:25:23 内存使用 0.32 MiB
显示代码纯文本
  1. #include<iostream>
  2. #include<cstdio>
  3. #include<algorithm>
  4. #include<cstring>
  5. #include<set>
  6. using namespace std;
  7. const int SIZEN=1010;
  8. int gcd(int a,int b){return !b?a:gcd(b,a%b);}
  9. class Point{
  10. public:
  11. int x,y;
  12. Point(int x_=0,int y_=0){x=x_,y=y_;}
  13. };
  14. Point normalize(Point a){
  15. int g=gcd(a.x,a.y);
  16. a.x/=g,a.y/=g;
  17. return a;
  18. }
  19. bool operator < (const Point &a,const Point &b){
  20. return a.x==b.x?a.y<b.y:a.x<b.x;
  21. }
  22. Point operator + (const Point &a,const Point &b){
  23. return Point(a.x+b.x,a.y+b.y);
  24. }
  25. Point operator - (const Point &a,const Point &b){
  26. return Point(a.x-b.x,a.y-b.y);
  27. }
  28. Point operator * (int a,const Point &b){
  29. return Point(a*b.x,a*b.y);
  30. }
  31. int operator * (const Point &a,const Point &b){
  32. return a.x*b.x+a.y*b.y;
  33. }
  34. int lensqr(const Point &a){
  35. return a*a;
  36. }
  37. Point vertical(const Point &a){//逆时针旋转90度
  38. return Point(-a.y,a.x);
  39. }
  40. class Line{
  41. public:
  42. Point pos;
  43. Point dir;//必须是normalize过的
  44. Line(){}
  45. Line(const Point &a,const Point &b){
  46. pos=a;
  47. dir=normalize(b-a);
  48. }
  49. };
  50. bool midper(const Point &a,const Point &b,Line &L){//ab中垂线为L,若不合法返回false
  51. L.pos=a+b;
  52. if(L.pos.x%2||L.pos.y%2) return false;
  53. L.pos.x/=2,L.pos.y/=2;
  54. L.dir=normalize(vertical(b-a));
  55. return true;
  56. }
  57. bool calc_sym(const Line &L,const Point &P,Point &P1){//对称点是否为整点,对称点存放在P1中
  58. //大致是先算出来P到L的垂足的参数
  59. int a=L.dir*L.dir,b=(P-L.pos)*L.dir;
  60. if(b%a) return false;
  61. int t=b/a;
  62. P1=2*(L.pos+t*L.dir)-P;
  63. return true;
  64. }
  65. int N;
  66. Point P[SIZEN];
  67. set<Point> PS;
  68. bool check(Line L){
  69. Point p1;
  70. for(int i=1;i<=N;i++){
  71. if(!calc_sym(L,P[i],p1)) return false;
  72. if(!PS.count(p1)) return false;
  73. }
  74. return true;
  75. }
  76. void work(void){
  77. int ans=0;
  78. //情况1:P1在对称轴外
  79. Line L;
  80. for(int i=2;i<=N;i++){
  81. if(midper(P[1],P[i],L)){
  82. if(check(L)) ans++;
  83. }
  84. }
  85. //情况2:P1在对称轴上,P2在对称轴外
  86. for(int i=3;i<=N;i++){
  87. if(midper(P[2],P[i],L)&&lensqr(P[i]-P[1])==lensqr(P[2]-P[1])){
  88. if(check(L)) ans++;
  89. }
  90. }
  91. //情况3:P1,P2均在对称轴上
  92. if(check(Line(P[1],P[2]))) ans++;
  93. printf("%d\n",ans);
  94. }
  95. void read(void){
  96. scanf("%d",&N);
  97. for(int i=1;i<=N;i++){
  98. scanf("%d%d",&P[i].x,&P[i].y);
  99. P[i]=2*P[i];
  100. PS.insert(P[i]);
  101. }
  102. }
  103. int main(){
  104. freopen("symmetry.in","r",stdin);
  105. freopen("symmetry.out","w",stdout);
  106. read();
  107. work();
  108. return 0;
  109. }