【BZOJ2648】SJY摆棋子(KD-Tree)
【BZOJ2648】SJY摆棋子(KD-Tree)
题面
Description
这天,SJY显得无聊。在家自己玩。在一个棋盘上,有N个黑色棋子。他每次要么放到棋盘上一个黑色棋子,要么放上一个白色棋子,如果是白色棋子,他会找出距离这个白色棋子最近的黑色棋子。此处的距离是 曼哈顿距离 即(|x1-x2|+|y1-y2|) 。现在给出N<=500000个初始棋子。和M<=500000个操作。对于每个白色棋子,输出距离这个白色棋子最近的黑色棋子的距离。同一个格子可能有多个棋子。
Input
第一行两个数 N M
以后M行,每行3个数 t x y
如果t=1 那么放下一个黑色棋子
如果t=2 那么放下一个白色棋子
Output
对于每个T=2 输出一个最小距离
Sample Input
2 3
1 1
2 3
2 1 2
1 3 3
2 4 2
Sample Output
1
2
题解
\(KD-Tree\)模板题?
对于分割出来的矩阵,维护矩阵中的点的两维坐标的最大和最小值
这样就可以计算当前点到达这个矩阵的最短距离
这样就可以剪枝啦
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 555555
#define ls (t[x].ch[0])
#define rs (t[x].ch[1])
#define cmin(a,b) (a>b?a=b:a)
#define cmax(a,b) (a<b?a=b:a)
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
int D;
int n,rt,Q;
struct Node{int d[2];}a[MAX];
bool operator<(Node a,Node b){return a.d[D]<b.d[D];}
struct KD_Tree_Node
{
int d[2],mn[2],mx[2];
int ch[2];
}t[MAX<<1];
int ans;
void update(int x,int y)
{
cmin(t[x].mn[0],t[y].mn[0]);cmin(t[x].mn[1],t[y].mn[1]);
cmax(t[x].mx[0],t[y].mx[0]);cmax(t[x].mx[1],t[y].mx[1]);
}
int Build(int l,int r,int nD)
{
int x=(l+r)>>1;D=nD;
nth_element(&a[l],&a[x],&a[r+1]);
t[x].d[0]=t[x].mn[0]=t[x].mx[0]=a[x].d[0];
t[x].d[1]=t[x].mn[1]=t[x].mx[1]=a[x].d[1];
if(l<x)ls=Build(l,x-1,nD^1),update(x,ls);
if(r>x)rs=Build(x+1,r,nD^1),update(x,rs);
return x;
}
int Dist(int nw,int X,int Y)
{
return max(t[nw].mn[0]-X,0)+max(X-t[nw].mx[0],0)+max(t[nw].mn[1]-Y,0)+max(Y-t[nw].mx[1],0);
}
void Query(int x,int xx,int yy)
{
int tmp=abs(t[x].d[0]-xx)+abs(t[x].d[1]-yy),d[2];
d[0]=ls?Dist(ls,xx,yy):2e9;
d[1]=rs?Dist(rs,xx,yy):2e9;
cmin(ans,tmp);tmp=d[0]>=d[1];
if(d[tmp]<ans)Query(t[x].ch[tmp],xx,yy);
if(d[tmp^1]<ans)Query(t[x].ch[tmp^1],xx,yy);
}
int main()
{
n=read();Q=read();
for(int i=1;i<=n;++i)a[i].d[0]=read(),a[i].d[1]=read();
rt=Build(1,n,0);
while(Q--)
{
int opt=read(),x=read(),y=read();
if(opt==1)
{
++n;
t[n].d[0]=t[n].mn[0]=t[n].mx[0]=x;
t[n].d[1]=t[n].mn[1]=t[n].mx[1]=y;
for(int nw=rt,d=0;nw;d^=1)
{
update(nw,n);
int k=t[n].d[d]>=t[nw].d[d];
if(!t[nw].ch[k]){t[nw].ch[k]=n;break;}
nw=t[nw].ch[k];
}
}
else
{
ans=2e9;Query(rt,x,y);
printf("%d\n",ans);
}
}
return 0;
}
上面是不会重建\(KD-Tree\)的代码
洛谷上数据有所加强,这样做会\(TLE\)
所以考虑替罪羊树类似的重构即可
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 555555
#define ls (t[x].ch[0])
#define rs (t[x].ch[1])
#define cmin(a,b) (a>b?a=b:a)
#define cmax(a,b) (a<b?a=b:a)
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
const double alpha=0.75;
int D;
int n,rt,m;
struct Node{int d[2];}a[MAX];
bool operator<(Node a,Node b){return a.d[D]<b.d[D];}
struct KD_Tree_Node
{
int d[2],mn[2],mx[2];
int ch[2],size;
}t[MAX<<1];
int ans;
int Q[MAX<<1],top,tot;
int Num;
int NewNode()
{
if(!top)return ++tot;
return Q[top--];
}
void update(int x,int y)
{
cmin(t[x].mn[0],t[y].mn[0]);cmin(t[x].mn[1],t[y].mn[1]);
cmax(t[x].mx[0],t[y].mx[0]);cmax(t[x].mx[1],t[y].mx[1]);
}
void pushup(int x)
{
t[x].mn[0]=t[x].mx[0]=t[x].d[0];
t[x].mn[1]=t[x].mx[1]=t[x].d[1];
t[x].size=t[ls].size+t[rs].size+1;
if(ls)update(x,ls);if(rs)update(x,rs);
}
int Build(int l,int r,int nD)
{
int mid=(l+r)>>1;D=nD;
nth_element(&a[l],&a[mid],&a[r+1]);
int x=NewNode();
t[x].d[0]=a[mid].d[0];t[x].d[1]=a[mid].d[1];
if(l<mid)ls=Build(l,mid-1,nD^1);else ls=0;
if(r>mid)rs=Build(mid+1,r,nD^1);else rs=0;
pushup(x);return x;
}
int Dist(int nw,int X,int Y)
{
return max(t[nw].mn[0]-X,0)+max(X-t[nw].mx[0],0)+max(t[nw].mn[1]-Y,0)+max(Y-t[nw].mx[1],0);
}
void Query(int x,int xx,int yy)
{
int tmp=abs(t[x].d[0]-xx)+abs(t[x].d[1]-yy),d[2];
d[0]=ls?Dist(ls,xx,yy):2e9;
d[1]=rs?Dist(rs,xx,yy):2e9;
cmin(ans,tmp);tmp=d[0]>=d[1];
if(d[tmp]<ans)Query(t[x].ch[tmp],xx,yy);
if(d[tmp^1]<ans)Query(t[x].ch[tmp^1],xx,yy);
}
void kill(int x){if(ls)kill(ls);a[++Num]=(Node){t[x].d[0],t[x].d[1]};Q[++top]=x;if(rs)kill(rs);}
void check(int &x,int nd)
{
if(t[x].size*alpha<t[ls].size||t[x].size*alpha<t[rs].size)
Num=0,kill(x),x=Build(1,Num,nd);
}
void insert(int &x,Node a,int nd)
{
if(!x){x=NewNode();t[x].d[0]=a.d[0];t[x].d[1]=a.d[1];pushup(x);return;}
if(a.d[nd]>t[x].d[nd])insert(rs,a,nd^1);
else insert(ls,a,nd^1);
pushup(x);check(x,nd);
}
int main()
{
n=read();m=read();
for(int i=1;i<=n;++i)a[i].d[0]=read(),a[i].d[1]=read();
rt=Build(1,n,0);
while(m--)
{
int opt=read(),x=read(),y=read();
if(opt==1)insert(rt,(Node){x,y},0);
else
{
ans=2e9;Query(rt,x,y);
printf("%d\n",ans);
}
}
return 0;
}
【BZOJ2648】SJY摆棋子(KD-Tree)的更多相关文章
- luogu4169 [Violet]天使玩偶/SJY摆棋子 / bzoj2648 SJY摆棋子 k-d tree
k-d tree + 重构的思想,就能卡过luogu和bzoj啦orz #include <algorithm> #include <iostream> #include &l ...
- BZOJ 2648: SJY摆棋子(K-D Tree)
Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 6051 Solved: 2113[Submit][Status][Discuss] Descript ...
- BZOJ2648: SJY摆棋子&&2716: [Violet 3]天使玩偶
BZOJ2648: SJY摆棋子 BZOJ2716: [Violet 3]天使玩偶 BZOJ氪金无极限... 其实这两道是同一题. 附上2648的题面: Description 这天,SJY显得无聊. ...
- [BZOJ2648] SJY摆棋子 kd-tree
2648: SJY摆棋子 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 5421 Solved: 1910[Submit][Status][Disc ...
- Bzoj2648 SJY摆棋子
Time Limit: 20 Sec Memory Limit: 128 MB Submit: 3128 Solved: 1067 Description 这天,SJY显得无聊.在家自己玩.在一个 ...
- bzoj 2648 SJY摆棋子 kd树
题目链接 初始的时候有一些棋子, 然后给两种操作, 一种是往上面放棋子. 另一种是给出一个棋子的位置, 问你离它最近的棋子的曼哈顿距离是多少. 写了指针版本的kd树, 感觉这个版本很好理解. #inc ...
- 2019.01.14 bzoj2648: SJY摆棋子(kd-tree)
传送门 kd−treekd-treekd−tree模板题. 题意简述:支持在平面上插入一个点,求对于一个点的最近点对. 思路:cdqcdqcdq是一种很不错的分治方法 只是好像码量有点窒息 所以我用了 ...
- BZOJ2648 SJY摆棋子(KD-Tree)
板子题. #include<iostream> #include<cstdio> #include<cmath> #include<cstdlib> # ...
- 【kd-tree】bzoj2648 SJY摆棋子
#include<cstdio> #include<cmath> #include<algorithm> using namespace std; #define ...
- [bzoj2648]SJY摆棋子(带插入kd-tree)
解题关键:带插入kdtree模板题. #include<iostream> #include<cstdio> #include<cstring> #include& ...
随机推荐
- zedboard学习第一篇
1. 刚开始学习使用,不知道从哪里开始,手上的资料也很乱,至于这个板子需要学什么也不清楚. 2. 第一个工程就从helloworld开始吧,Zed板上的Zynq是一个PS(processing sys ...
- 卸载Oracle 11g
卸载oracle只需要执行deinstall.ba即可 运行.bat文件 按照提示步骤进行卸载 .bat文件所在目录需要手动删除,需要在程序管理器里面关掉Oracle的进程,在删除当前文件夹
- Net Core学习笔记
Net Core 官网:https://dotnet.github.io/ Net Core Api: https://docs.microsoft.com/en-us/dotnet/api/?vie ...
- Qt官方开发环境生成的exe发布方式
本来想自己写一个打包程序的文章了,但是我发现了宝贝,在这里,大神写的比我牛逼的多了,这里做一下搬运工 一是为了方便大家 二是为了以后方便自己找 原文链接:http://tieba.baidu.com/ ...
- 【xmlHttp_Class 远程访问类】使用说明
类名:xmlHttp_Class 说明:远程获取外部网站数据信息或执行一个外部网站程序 目录: 类型 名称 参数 返回 说明 属性 [必需] [xmlHttp].url = [urlString] - ...
- 浅谈java中接口与抽象类之间的异同
刚学习java的时候,总觉得接口和抽象类很像,但又说不上具体有什么区别.今天静下来,翻翻书,查查资料,做个小结.首先举两个例子,看看interface和abstract class 在“外形”上有啥异 ...
- 骰子涂色 (Cube painting,UVa 253)
题目描述:算法竞赛入门习题4-4 题目思路:1.旋转其中一个骰子进行匹配 2.进行遍历,如果匹配,就进行相对面的匹配 3.三个对立面都匹配即是一样等价的 //没有按照原题的输入输出 #include ...
- LeetCode 94 ——二叉树的中序遍历
1. 题目 2. 解答 2.1. 递归法 定义一个存放树中数据的向量 data,从根节点开始,如果节点不为空,那么 递归得到其左子树的数据向量 temp,将 temp 合并到 data 中去 将当前节 ...
- Image控件显示以byte[]字节数组形式存在的图片
工作中遇到了这样的一个问题.起初觉得很简单,获得了图片的byte[]后,可以将其转换成内存中的图片对象(如System.Drawing.Image),而后赋给页面的Image控件.尝试后才发现这样根本 ...
- selenium中的三种等待方式(显示等待WebDriverWait()、隐式等待implicitly()、强制等待sleep())---基于python
我们在实际使用selenium或者appium时,等待下个等待定位的元素出现,特别是web端加载的过程,都需要用到等待,而等待方式的设置是保证脚本稳定有效运行的一个非常重要的手段,在selenium中 ...