题意 给出n个二维点(2e5) 和 q个询问(1e4) 每个询问给lr 问点l到r间有多少对点的曼哈顿距离<=d 点的坐标<=108

想出了莫队算法 复杂度n^1.5 看起来很科学 但是每次del和add点是直接进行暴力扫当前LR区间的点来算 ans将要增加/减少多少 实际这样的复杂度是V^2.5 会t掉

由于是在二维上找一定范围的点的数目 就可以想到KDTree V^1.5log(n)在15s下还是很科学的

做了一些曼哈顿距离的getdis函数 调完又t了。。

我使用了一下青岛的K的写法 开出三个维度 x y id 每次询问的是 和x1y1曼哈顿距离<=d 并且 L <= id <= R 的点 因为曼哈顿不是一个矩形 是一个正方形旋转了的图形 搜起来会比较麻烦

但是我感觉这个就是正解。。在网上也百度不到题解 于是出去问了一下别人对这个题的看法 然而也是莫队+KDT做的 就很迷茫

要来代码一看 有一个和我不同的是这样的 把输入进去的xy 都变成x+y 和 x-y 在查询的时候不围绕改变后的xy做一个曼哈顿的查询 而是直接查询矩形

xl = x - d   xr = x + d   yl = y - d   yr = y + d  这样

查了查对这个优化的意义 这个优化的意思 是 把曼哈顿距离转化为切比雪夫距离

http://m.blog.csdn.net/article/details?id=47259831 这个里面讲的

旋转之前两个点的曼哈顿距离是d 那么旋转后两个点的切比雪夫距离也是d 所以由一个点就可以求出来一个矩阵出来 KDT跑矩阵很快

还有一个点 就是KDT的删除点

KDT的插入点是很简单的logn操作 但是删除点的话 由于这个点可能管辖一定的点 删除了它就需要从它的孩子中找出一个点来管辖这个区域 相当于每次删除都重构

其实不需要这样 我们可以先把这棵树建出来 一开始这个树只有形状 但是点都没有放进去 每次放进去一个点就相当于点亮一个点

可以做一个类似于离线的操作 我们现在已经知道哪些点会被插入进这个树了 只要它进去 我们就记录下来

然后用这些点的形状建树 虽然一开始没有点被插进去

我们在结构体里面定义val代表当前的管辖点存在与否/有多少个 sum代表这个区域有多少个 fa代表这个点的父亲节点是谁

所以每次删除或者插入一个点 只需要改掉这个点的val 然后把这个点及其先辈的sum改了就可以了 由于我们事先知道了所有的点 所以建出来的树深度还是logn的

这种KDT写法也是很优雅的。。

#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<math.h>
#include<map>
#include<string>
#include<vector>
#include<queue>
#include<stack>
#include<iostream>
using namespace std;
#define pb push_back int n , d , q ;
int m , root , cmp_d ;
int xl , xr , yl , yr ;
int L , R ; struct node {
int d[3] , Max[3] , Min[3] ;
int sum ;
int fa ;
int val ;
int l , r ;
int id ;
}tr[200050];
bool cmp(node a , node b) {
return a.d[cmp_d] < b.d[cmp_d] ;
}
int ix[200050] , iy[200050] ; int pos[200050] ;
struct no {
int l , r ;
int id ;
}xw[10050];
bool cmp2(no a , no b) {
if(pos[a.l] == pos[b.l]) return a.r < b.r ;
return a.l < b.l ;
}
int an ;
int ans[10050] ;
void up(int p , int k) {
for(int i=0;i<2;i++) {
if(tr[p].Max[i] < tr[k].Max[i]) tr[p].Max[i] = tr[k].Max[i] ;
if(tr[p].Min[i] > tr[k].Min[i]) tr[p].Min[i] = tr[k].Min[i] ;
}
tr[p].sum += tr[k].sum ;
}
int w[200050] ;
int build(int l , int r , int D , int fa) {
int mid = (l + r) >> 1 ;
cmp_d = D ;
nth_element(tr+l+1,tr+mid+1,tr+r+1,cmp) ;
w[tr[mid].id] = mid ;
tr[mid].fa = fa ;
tr[mid].val = 0 ;
tr[mid].sum = 0 ;
for(int i = 0 ; i < 2 ; i ++ ) tr[mid].Max[i] = tr[mid].Min[i] = tr[mid].d[i] ;
if(l != mid) tr[mid].l = build(l , mid-1 , D^1 , mid) ; else tr[mid].l = 0 ;
if(r != mid) tr[mid].r = build(mid+1 , r , D^1 , mid) ; else tr[mid].r = 0 ;
if(tr[mid].l) up(mid , tr[mid].l) ;
if(tr[mid].r) up(mid , tr[mid].r) ;
return mid ;
}
void upda(int rt , int val) {
tr[rt].val += val ;
while(rt) {
tr[rt].sum += val ;
rt = tr[rt].fa ;
}
}
int query(int p) {
if(tr[p].sum == 0) return 0 ;
if(tr[p].Max[0] < xl || tr[p].Min[0] > xr || tr[p].Max[1] < yl || tr[p].Min[1] > yr) return 0 ;
if(tr[p].Max[0] <= xr && tr[p].Min[0] >= xl && tr[p].Max[1] <= yr && tr[p].Min[1] >= yl) return tr[p].sum ;
int ret = 0 ;
if(tr[p].d[0] <= xr && tr[p].d[0] >= xl && tr[p].d[1] <= yr && tr[p].d[1] >= yl) ret += tr[p].val ;
if(tr[p].l) ret += query(tr[p].l) ;
if(tr[p].r) ret += query(tr[p].r) ;
return ret ;
}
void cxh(int x , int y) {
xl = x - d ; xr = x + d ;
yl = y - d ; yr = y + d ;
} int main () {
int cas = 1 ;
while(scanf("%d%d%d" , &n , &d , &q) != EOF) {
int sz = sqrt(n) ;
for(int i = 1 ; i <= n ; i ++ ) {
scanf("%d%d" , &ix[i] , &iy[i]) ;
pos[i] = i / sz ;
tr[i].l = tr[i].r = 0 ;
tr[i].d[0] = ix[i] + iy[i] ;
tr[i].d[1] = ix[i] - iy[i] ; int z = ix[i] ;
ix[i] = ix[i] + iy[i] ;
iy[i] = z - iy[i] ; tr[i].id = i ;
}
root = build(1 , n , 0 , 0) ;
an = 0 ;
for(int i = 1 ; i <= q ; i ++ ) {
scanf("%d%d" , &xw[i].l , &xw[i].r) ;
xw[i].id = i ;
}
L = 0 ;
R = 0 ;
sort(xw+1,xw+1+q,cmp2) ;
for(int i = 1 ; i <= q ; i ++ ) {
while(L < xw[i].l) {
cxh(ix[L] , iy[L]) ;
upda(w[L] , -1) ;
L ++ ;
an -= query(root) ;
}
while(L > xw[i].l) {
L -- ;
cxh(ix[L] , iy[L]) ;
an += query(root) ;
upda(w[L] , 1) ;
}
while(R < xw[i].r) {
R ++ ;
cxh(ix[R] , iy[R]) ;
an += query(root) ;
upda(w[R] , 1) ;
}
while(R > xw[i].r) {
cxh(ix[R] , iy[R]) ;
upda(w[R] , -1) ;
R -- ;
an -= query(root) ;
}
ans[xw[i].id] = an ;
}
printf("Case %d:\n" , cas ++ ) ;
for(int i = 1 ; i <= q ; i ++ ) printf("%d\n" , ans[i]) ;
}
}

CSU 1786 莫队+KDTree的更多相关文章

  1. CSU 1515 Sequence (莫队算法)

    题意:给n个数,m个询问.每个询问是一个区间,求区间内差的绝对值为1的数对数. 题解:先离散化,然后莫队算法.莫队是离线算法,先按按询问左端点排序,在按右端点排序. ps:第一次写莫队,表示挺简单的, ...

  2. BZOJ 3289: Mato的文件管理[莫队算法 树状数组]

    3289: Mato的文件管理 Time Limit: 40 Sec  Memory Limit: 128 MBSubmit: 2399  Solved: 988[Submit][Status][Di ...

  3. NBUT 1457 莫队算法 离散化

    Sona Time Limit:5000MS     Memory Limit:65535KB     64bit IO Format: Submit Status Practice NBUT 145 ...

  4. 【填坑向】bzoj2038小Z的袜子 莫队

    学莫队必做题,,,但是懒得写.今天来填个坑 莫队水题 莫队实际上就是按一个玄学顺序来离线计算询问,保证复杂度只会多一个n1/2,感觉是玄学(离线算法都很玄学) 易错点:要开long long(卡我半天 ...

  5. BZOJ 2038: [2009国家集训队]小Z的袜子(hose) [莫队算法]【学习笔记】

    2038: [2009国家集训队]小Z的袜子(hose) Time Limit: 20 Sec  Memory Limit: 259 MBSubmit: 7687  Solved: 3516[Subm ...

  6. NPY and girls-HDU5145莫队算法

    Time Limit: 8000/4000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Problem Description ...

  7. Codeforces617 E . XOR and Favorite Number(莫队算法)

    XOR and Favorite Number time limit per test: 4 seconds memory limit per test: 256 megabytes input: s ...

  8. Bzoj 2038---[2009国家集训队]小Z的袜子(hose) 莫队算法

    题目链接 http://www.lydsy.com/JudgeOnline/problem.php?id=2038 Description 作为一个生活散漫的人,小Z每天早上都要耗费很久从一堆五颜六色 ...

  9. 【BZOJ 3735】苹果树 树上莫队(树分块+离线莫队+鬼畜的压行)

    2016-05-09 UPD:学习了新的DFS序列分块,然后发现这个东西是战术核导弹?反正比下面的树分块不知道要快到哪里去了 #include<cmath> #include<cst ...

随机推荐

  1. Assertion (software development) -- 断言

    Wiki: In computer programming, an assertion is a predicate (a true–false statement) placed in a prog ...

  2. 1366 贫富差距(floyed)

    1366 贫富差距 题目来源: TopCoder 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题 一个国家有N个公民,标记为0,1,2,...,N-1,每个公民有一 ...

  3. 常见的.NET面试题(130)

    1. 简述 private. protected. public. internal 修饰符的访问权限. 答 . private : 私有成员, 在类的内部才可以访问. protected : 保护成 ...

  4. NetCore

    正确理解DTO.值对象和POCO https://www.cnblogs.com/redmoon/archive/2015/04/29/4467485.html ASP.NET Core 配置系统 h ...

  5. Sublime Text 3如何快速生成HTML5的头部信息和常用的快捷键

    一.快速生成HTML5的头部信息的步骤: 1.Ctrl + N,新建一个文档: 2.Ctrl + Shift + P,打开命令模式,再输入 sshtml 进行模糊匹配,将语法切换到html模式: 3. ...

  6. js,jquery和dojo操作dom

    最近想学习arcgis javascript api,拦路虎就是dojo,为了便于理解dojo,在学习dojo的同时参考原生js和jquery,下午学习了下dom操作,mark下! 一.获取元素 js ...

  7. mybatis springmvc velocity的demo

    https://codeload.github.com/xingfly/SBlog/zip/master 1.记得导入数据库并修改 数据库的配置. 2.使用jetty运行 pom 添加jetty &l ...

  8. 0x04 MySQl 表操作

    0x01 存储引擎介绍 存储引擎即表类型,mysql根据不同的表类型会有不同的处理机制 详见:http://www.cnblogs.com/linhaifeng/articles/7213670.ht ...

  9. 什么是lambda函数?它有什么好处?

    lambda 函数是一个可以接收任意多个参数(包括可选参数)并且返回单个表达式值的函数. lambda 函数不能包含命令,它们所包含的表达式不能超过一个.不要试图向lambda 函数中塞入太多的东西: ...

  10. 常用算法及其Python实现

    冒泡排序 ​ def bubble_sort(li):    for i in range(len(li)-1): # i表示第几趟        for j in range(len(li)-i-1 ...