不会KD-tree怎么办?CQD硬搞。

建立正常的平面直角坐标系,首先我们只考虑在目标点左下角的点对目标点的贡献,由于左下点的横纵坐标都小于目标点,那么曼哈顿距离就可以化简了,绝对值去掉后,得到$x2+y2-(x1+y1)$,那么我们的目标就转化为了求横纵坐标以及时间轴都小于目标查询点的更改点所作出的贡献,这是一个三维偏序问题,我们在树状数组中维护x+y的最大值,进而即可更新答案。

可是这样做我们只是考虑了左下角点的贡献,肯定是会出错的,但是其余位置的点不容易化简绝对值,或者化简完以后的形式比较难以维护,而且个人认为分类的话码量略大。

那么我们换一个角度,如何将其余位置的点都变化到目标点的左下角。

直接翻转坐标系就行了,把所有点的横纵坐标都翻转一下,使之分别落于其他象限。那么再次调用cdq即可,注意防负下标。

这样打的好处就是比较无脑,而且正确性保障很大,但是常数这种东西还是很神奇的。(本代码不保证Bzoj可A)(其实应该不可A,因为main函数返回值是signed)

#include<bits/stdc++.h>
#define lowbit(x) (x&(-x))
#define int long long
using namespace std;
const int inf=0x7fffffffffffff;
int read(){
int sum=,f=;char x=getchar();
while(x<''||x>''){
if(x=='-') f=-;
x=getchar();
}while(x>=''&&x<=''){
sum=sum*+x-'';
x=getchar();
}return sum*f;
}
struct rec{
int x,y,t,id,ty;
friend bool operator < (const rec &a,const rec &b){
if(a.t==b.t){
if(a.x==b.x) return a.y<b.y;
else return a.x<b.x;
}else return a.t<b.t;
}
}q[],tmp[];
int n,m,kind,Max,ans[],tr[],_time=,mk[];
void change(int pos,int val){
if(!pos) return ;
for(int i=pos;i<=Max;i+=lowbit(i))
if(mk[i]!=_time){
mk[i]=_time;
tr[i]=val;
}
else tr[i]=max(tr[i],val);
}
int ask(int pos){
int ans=-inf;
for(int i=pos;i;i-=lowbit(i))
if(mk[i]==_time) ans=max(ans,tr[i]);
return ans;
}
void cdq(int l,int r){
if(l==r) return ;
int mid=l+r>>;
cdq(l,mid);cdq(mid+,r);
int i=l,j=mid+,tot=l;
while(i<=mid&&j<=r){
if(q[i].x<=q[j].x){
tmp[tot]=q[i];
if(!q[i].ty)
change(q[i].y,q[i].x+q[i].y);
++tot;++i;
}else {
tmp[tot]=q[j];
if(q[j].ty)
ans[q[j].id]=min(ans[q[j].id],q[j].x+q[j].y-ask(q[j].y));
++tot;++j;
}
}
while(i<=mid){
tmp[tot]=q[i];
// if(!q[i].ty)
// change(q[i].y,q[i].x+q[i].y);
++tot;++i;
}
while(j<=r){
tmp[tot]=q[j];
if(q[j].ty)
ans[q[j].id]=min(ans[q[j].id],q[j].x+q[j].y-ask(q[j].y));
++tot;++j;
}
/* for(int i=1;i<=Max;i++)
cout<<tr[i]<<" ";cout<<endl;*/
// for(int k=l;k<=mid;k++)
// if(!q[i].ty) del(q[k].y);
/* for(int i=1;i<=Max;i++)
cout<<tr[i]<<" ";cout<<endl;*/
_time++;
// if(_time>=10000000000) _time=1;
for(int k=l;k<=r;k++) q[k]=tmp[k];
}
signed main(){
n=read();m=read();
for(int i=;i<=n;i++){
q[i].x=read()+;q[i].y=read()+;
q[i].t=;q[i].ty=;
Max=max(Max,max(q[i].x,q[i].y));
}
for(int i=;i<=m;i++){
kind=read();
if(kind&){
q[i+n].x=read()+;q[i+n].y=read()+;
q[i+n].t=i;q[i].ty=;
}else {
q[i+n].x=read()+;q[i+n].y=read()+;
q[i+n].t=i;q[i+n].ty=;
q[i+n].id=++ans[];
ans[ans[]]=inf;
}
Max=max(Max,max(q[i+n].x,q[i+n].y));
}n+=m;
// cout<<"Max="<<Max<<endl;
sort(q+,q+n+);
cdq(,n);
/* for(int i=1;i<=n;i++)
cout<<q[i].t<<" "<<q[i].x<<" "<<q[i].y<<endl;*/
for(int i=;i<=n;i++) q[i].x=-q[i].x+Max+;
sort(q+,q++n);
/* for(int i=1;i<=n;i++)
cout<<q[i].t<<" "<<q[i].x<<" "<<q[i].y<<endl;*/
cdq(,n);
for(int i=;i<=n;i++) q[i].y=-q[i].y+Max+;
sort(q+,q++n);
cdq(,n);
for(int i=;i<=n;i++) q[i].x=-q[i].x+Max+;
sort(q+,q+n+);
cdq(,n);
for(int i=;i<=ans[];i++)
printf("%lld\n",ans[i]);
return ;
}

BZOJ2716天使玩偶的更多相关文章

  1. [BZOJ2716]天使玩偶

    [BZOJ2716]天使玩偶 题目大意: 一个平面直角坐标系,坐标\(1\le x,y\le10^6\).\(n(n\le10^6)\)次操作,操作包含以下两种: 新增一个点\((x,y)\): 询问 ...

  2. [BZOJ2716] [Violet 3]天使玩偶(CDQ分治)

    [BZOJ2716] [Violet 3]天使玩偶(CDQ分治) 题面 Ayu 在七年前曾经收到过一个天使玩偶,当时她把它当作时间囊埋在了地下.而七年后 的今天,Ayu 却忘了她把天使玩偶埋在了哪里, ...

  3. bzoj2648SJY摆棋子&&bzoj2716[Violet 3]天使玩偶*

    bzoj2648SJY摆棋子 bzoj2716[Violet 3]天使玩偶 题意: 棋盘上有n个棋子,现在有m个操作,一种是加棋子,一种是查询离某个点最近的棋子.n,m≤500000. 题解: 先将已 ...

  4. bzoj2716/2648 / P4169 [Violet]天使玩偶/SJY摆棋子

    P4169 [Violet]天使玩偶/SJY摆棋子 k-d tree 模板 找了好几天才发现输出优化错了....真是zz...... 当子树非常不平衡时,就用替罪羊树的思想,拍扁重建. luogu有个 ...

  5. BZOJ2648: SJY摆棋子&&2716: [Violet 3]天使玩偶

    BZOJ2648: SJY摆棋子 BZOJ2716: [Violet 3]天使玩偶 BZOJ氪金无极限... 其实这两道是同一题. 附上2648的题面: Description 这天,SJY显得无聊. ...

  6. BZOJ 2716: [Violet 3]天使玩偶

    2716: [Violet 3]天使玩偶 Time Limit: 80 Sec  Memory Limit: 128 MBSubmit: 1473  Solved: 621[Submit][Statu ...

  7. 【BZOJ-2648&2716】SJY摆棋子&天使玩偶 KD Tree

    2648: SJY摆棋子 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 2459  Solved: 834[Submit][Status][Discu ...

  8. BZOJ 2716: [Violet 3]天使玩偶( CDQ分治 + 树状数组 )

    先cdq分治, 然后要处理点对答案的贡献, 可以以询问点为中心分成4个区域, 然后去掉绝对值(4种情况讨论), 用BIT维护就行了. --------------------------------- ...

  9. CH4701 天使玩偶

    题意 4701 天使玩偶 0x40「数据结构进阶」例题 描述 题目PDF 样例输入 2 3 1 1 2 3 2 1 2 1 3 3 2 4 2 样例输出 1 2 来源 石家庄二中Violet 3杯省选 ...

随机推荐

  1. axios配置及使用(发起请求时带上token)

    1.安装 利用npm安装 npm install axios --save 2.引入即可使用 import axios from 'axios' 3.目录 4.各个文件设置: (1)env.js ex ...

  2. 1+X证书学习日志——盒模型

    ##   padding的作用:                 控制子元素和父元素之间的位置关系                              padding设置方法:          ...

  3. 多个div并排不换行

    1.所有div的父元素不换行   white-space: nowrap; 2.所有div设置为行内元素  display: inline-block; 基于java记账管理系统[尚学堂·百战程序员]

  4. c语言二进制、八进制、十六进制

    int binary = 0b01000010; //二进制 printf("%d\n", binary); //十进制 printf("0x%x\n", 0x ...

  5. python 复制列表

    python的变量仅仅是指向对象的标签,所以在操作列表的时候,list1 = list2这种做法只会复制一个标签,然后指向对象,并非生成一个新的对象. 大致有5中方法可以复制列表: a = [1,2, ...

  6. Computer Vision_33_SIFT:An efficient SIFT-based mode-seeking algorithm for sub-pixel registration of remotely sensed images——2015

    此部分是计算机视觉部分,主要侧重在底层特征提取,视频分析,跟踪,目标检测和识别方面等方面.对于自己不太熟悉的领域比如摄像机标定和立体视觉,仅仅列出上google上引用次数比较多的文献.有一些刚刚出版的 ...

  7. 分布式session的几种实现方式

    在搭建完集群环境后,不得不考虑的一个问题就是用户访问产生的session如何处理.如果不做任何处理的话,用户将出现频繁登录的现象,比如集群中存在A.B两台服务器,用户在第一次访问网站时,Nginx通过 ...

  8. servlet版本与tomcat版本对应关系,各版本web.xml头信息写法

    The mapping between the specifications and the respective Apache Tomcat versions is: Servlet Spec JS ...

  9. RT-Thread--内核移植

    内核移植 内核移植就是指将 RT-Thread 内核在不同的芯片架构.不同的板卡上运行起来,能够具备线程管理和调度,内存管理,线程间同步和通信.定时器管理等功能.移植可分为 CPU 架构移植和 BSP ...

  10. js、jquery实现列表模糊搜索功能

    实现的搜索功能: 1. 可以匹配输入的字符串找出列表中匹配的项,列表框的高度跟随搜索出的列表项的多少改变 2. 可以点击某一项进行选中列表项 3. 可以按下上.下.回车键来控制列表项 4. 按下回车键 ...