首页 > 代码库 > codeforces 425D
codeforces 425D
题意:给定n<=100000个二维点,并且0<=x,y<=100000,求有多少个平行于坐标轴的正方形
思路:本来想hash的,但是感觉不好弄。。
后来感觉像是分块,最坏的情况就是那种x,y点稠密在一起的情况,并且x与y大致相同的情况下答案最多。。
然后就想到了跟分块很像的暴力。
对于每个点,用vx[x]记录有p.x==x的点的y,用vy[y]记录p.y==y的x
那么对于每个vx[], vy[]排序。。
接着对于每个点我们统计以它为右上角的正方形。。
那么对于每个点(x, y),我们二分查找x处于vy[y]的位置ty,我们二分查找y处于vx[x]的位置tx
那么如果tx<ty,我们直接统计vx[x][0]~vx[x][tx-1]的所有y是否有符合的
每次判断可以通过使用二分查找其他点是否存在。。
如果tx>=ty则统计vy[y][0]~vy[y][ty-1]的所有y是否符合。方法类似。
这样最坏情况下应该跟分块的复杂度差不多吧(不会证明)
code:
1 #include <bits/stdc++.h> 2 #define M0(x) memset(x, 0, sizeof(x)) 3 #define MP make_pair 4 #define PB push_back 5 #define repf(i, a, b) for (int i = (a); i <= (b); ++i) 6 using namespace std; 7 #define x first 8 #define y second 9 const int maxn = 101010;10 pair<int, int> p[maxn];11 vector<int> vx[maxn], vy[maxn];12 int n, m;13 14 void solve(){15 int mx = 0, my = 0;16 repf(i, 1, n){17 scanf("%d%d", &p[i].x, &p[i].y);18 mx = max(p[i].x, mx);19 my = max(p[i].y, my);20 }21 repf(i, 0, mx) vx[i].clear();22 repf(i, 0, my) vy[i].clear();23 repf(i, 1, n){24 vx[p[i].x].PB(p[i].y);25 vy[p[i].y].PB(p[i].x);26 }27 repf(i, 0, mx) sort(vx[i].begin(), vx[i].end());28 repf(i, 0, my) sort(vy[i].begin(), vy[i].end());29 int ans = 0;30 int tx, ty, x, y, t2, t3, d, nx, ny;31 repf(i, 1, n){32 x = p[i].x, y = p[i].y;33 tx = lower_bound(vx[x].begin(), vx[x].end(), y) - vx[x].begin();34 ty = lower_bound(vy[y].begin(), vy[y].end(), x) - vy[y].begin();35 if (tx < ty){36 t2 = ty;37 for (int j = tx-1; j >= 0; --j){38 d = y - vx[x][j];39 nx = x - d;40 t2 = lower_bound(vy[y].begin(), vy[y].end(), nx) - vy[y].begin();41 if (vy[y][t2] == nx){42 t3 = lower_bound(vx[nx].begin(), vx[nx].end(), vx[x][j]) - vx[nx].begin();43 if (vx[nx][t3] == vx[x][j]) ++ans;44 }45 }46 continue;47 }48 // t2 = tx;49 for (int j = ty-1; j >= 0; --j){50 d = x - vy[y][j];51 ny = y - d;52 t2 = lower_bound(vx[x].begin(), vx[x].end(), ny) - vx[x].begin();53 if (vx[x][t2] == ny){54 t3 = lower_bound(vy[ny].begin(), vy[ny].end(), vy[y][j]) - vy[ny].begin();55 if (vy[ny][t3] == vy[y][j]) ++ans;56 }57 }58 }59 cout << ans << endl; 60 }61 62 int main(){63 // freopen("a.in", "r", stdin);64 // freopen("a.out", "w", stdout);65 while (scanf("%d", &n) != EOF){66 solve();67 }68 return 0;69 }
codeforces 425D
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。