zoukankan      html  css  js  c++  java
  • POJ 2002 几何+hash

     题目大意:

    给定1000个点,寻找有多少组四点对能组成正方形

    这里的题目跟上一道做的找平行四边形类似但想法却又不相同的方法

    这里找任意2个点形成的一条边,那么可以根据这两个点,找到能和他们组成正方形剩下的两个点的位置,根据hash表去搜索,如果这两个位置存在自己需要的点,说明这种方案可行

    添加查找均交给hash表,这样可以实现O(n*n)的复杂度

    最后因为每个正方形因为有4条边被访问了4次,所以总的答案要除以4

     1 #include <cstdio>
     2 #include <cstring>
     3 #include <iostream>
     4 #include <algorithm>
     5 
     6 using namespace std;
     7 #define N 1010
     8 #define MOD 1000007
     9 int head[MOD+2] , k;
    10 
    11 struct Point{
    12     int x , y;
    13     Point(int x=0 , int y=0):x(x),y(y){}
    14     bool operator==(const Point &m)const{
    15         return x==m.x&&y==m.y;
    16     }
    17     void input(){scanf("%d%d" , &x , &y);}
    18     void print(){cout<<x<<" "<<y<<endl;}
    19 }p[N];
    20 
    21 struct HashNode{
    22     int id , next;
    23 }_hash[N];
    24 
    25 void insert(int i)
    26 {
    27     int v = (p[i].x*p[i].x+p[i].y*p[i].y)%MOD;
    28     _hash[k].id = i , _hash[k].next = head[v];
    29     head[v] = k++;
    30 }
    31 
    32 bool find(Point a)
    33 {
    34     int v = (a.x*a.x+a.y*a.y)%MOD;
    35     for(int i=head[v] ; ~i ; i=_hash[i].next){
    36         if(p[_hash[i].id] == a) return true;
    37     }
    38     return false;
    39 }
    40 
    41 int main()
    42 {
    43     #ifndef ONLINE_JUDGE
    44         freopen("a.in" , "r" , stdin);
    45     #endif // ONLINE_JUDGE
    46     int n;
    47     while(scanf("%d" , &n) , n)
    48     {
    49         memset(head , -1 , sizeof(head));
    50         k = 0;
    51         for(int i=1 ; i<=n ; i++){
    52             p[i].input();
    53             insert(i);
    54         }
    55         int ret = 0;
    56         for(int i=1 ; i<=n ; i++){
    57             for(int j=i+1 ; j<=n ; j++){
    58                 int dy = p[j].y-p[i].y;
    59                 int dx = p[j].x-p[i].x;
    60                 Point p1 = Point(p[i].x+dy , p[i].y-dx);
    61                 Point p2 = Point(p[j].x+dy , p[j].y-dx);
    62                 if(find(p1) && find(p2)) ret++;
    63                 p1 = Point(p[i].x-dy , p[i].y+dx);
    64                 p2 = Point(p[j].x-dy , p[j].y+dx);
    65                 if(find(p1) && find(p2)) ret++;
    66             }
    67         }
    68         printf("%d
    " , ret/4);
    69     }
    70 }
  • 相关阅读:
    gcc编译常用选项
    汇编指令随笔
    VIM简单操作
    Linux编译阻塞型驱动遇到'TASK_NORMAL' undeclared (first use in this function)问题解决办法
    Linux中断流程分析
    Linux驱动开发之LED驱动
    函数(三)
    函数(二)
    文件和函数(一)
    文件处理和字符编码
  • 原文地址:https://www.cnblogs.com/CSU3901130321/p/4544740.html
Copyright © 2011-2022 走看看