http://www.lydsy.com/JudgeOnline/problem.php?id=2648 (题目链接)

题意

  动态维护二维平面上的点的插入以及最邻近域搜索。

Solution

  KDtree板子,代码膜的XlightGod。复杂度真的萎,感觉主要还是剪枝。

  理论:http://blog.csdn.net/silangquan/article/details/41483689

  实现:http://blog.xlightgod.com/%E3%80%90bzoj2648%E3%80%91sjy%E6%91%86%E6%A3%8B%E5%AD%90/

细节

  码农数据结构题

代码

#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<cmath>
#define LL long long
#define inf 2147483640
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
using namespace std; const int maxn=1000010;
int n,rt,K,m,ans; struct node {
int v[2],mn[2],mx[2],son[2];
friend bool operator < (const node &a,const node &b) {return a.v[K]<b.v[K];}
int& operator [] (int x) {return son[x];}
}tr[maxn],S; void pushup(int k) {
for (int i=0;i<2;i++) {
if (!tr[k][i]) continue;
for (int j=0;j<2;j++) {
tr[k].mx[j]=max(tr[k].mx[j],tr[tr[k][i]].mx[j]);
tr[k].mn[j]=min(tr[k].mn[j],tr[tr[k][i]].mn[j]);
}
}
}
int build(int l,int r,int p) {
K=p;int mid=(l+r)>>1;
nth_element(tr+l,tr+mid,tr+r+1); //闭,闭,开
for (int i=0;i<2;i++) tr[mid].mn[i]=tr[mid].mx[i]=tr[mid].v[i];
if (l<mid) tr[mid][0]=build(l,mid-1,p^1);
if (r>mid) tr[mid][1]=build(mid+1,r,p^1);
pushup(mid);
return mid;
}
void insert(int &k,int p) {
K=p;
if (!k) {
k=++n;tr[k]=S;
for (int i=0;i<2;i++) tr[k].mx[i]=tr[k].mn[i]=tr[k].v[i];
return;
}
else insert(tr[k][S<tr[k]],p^1);
pushup(k);
}
int dis(node a,node b) {
int res=0;
for (int i=0;i<2;i++) res+=abs(a.v[i]-b.v[i]);
return res;
}
int eva(int k) { //类似估价函数
if (!k) return inf;
int res=0;
for (int i=0;i<2;i++) res+=max(0,tr[k].mn[i]-S.v[i])+max(0,S.v[i]-tr[k].mx[i]);
return res;
}
void query(int k) {
ans=min(ans,dis(S,tr[k]));
int dl=eva(tr[k][0]),dr=eva(tr[k][1]);
if (dl<dr) {
if (dl<ans) query(tr[k][0]);
if (dr<ans) query(tr[k][1]);
}
else {
if (dr<ans) query(tr[k][1]);
if (dl<ans) query(tr[k][0]);
}
}
int main() {
scanf("%d%d",&n,&m);
for (int i=1;i<=n;i++) scanf("%d%d",&tr[i].v[0],&tr[i].v[1]);
rt=build(1,n,0);
for (int op,i=1;i<=m;i++) {
scanf("%d%d%d",&op,&S.v[0],&S.v[1]);
if (op==1) insert(rt,0);
else {
ans=inf;
query(rt);
printf("%d\n",ans);
}
}
return 0;
}

Solution

  CDQ分治。考虑在查询点左下方的点里面$|x+y|$最大的,这个每次CDQ分治的时候按照x排序然后树状数组维护一下就可以了。然后我们将坐标轴旋转一下,做4个方向,然后就搞完了。

  TTTTTLE,过不了了,也许你需要底层优化。

细节

  清空的时候注意复杂度。

代码

// bzoj2648
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<cmath>
#define LL long long
#define inf 2147483600
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout)
using namespace std;
inline int gi() {
int x=0,f=1;char ch=getchar();
while (ch<'0' || ch>'9') {if (ch=='-') f=-1;ch=getchar();}
while (ch>='0' && ch<='9') {x=x*10+ch-'0';ch=getchar();}
return x*f;
} const int maxn=1000010;
int n,m,X,Y,c[maxn];
struct data {int x,y,id,op,ans;}q[maxn],nq[maxn]; bool cmp(data a,data b) {return a.x<b.x;}
bool cmpid(data a,data b) {return a.id<b.id;}
int lowbit(int x) {
return x&-x;
}
void modify(int x,int val) {
for (int i=x;i<=Y;i+=lowbit(i)) c[i]=max(c[i],val);
}
void clear(int x) {
for (int i=x;i<=Y;i+=lowbit(i)) c[i]=-inf;
}
int query(int x) {
int res=-inf;
for (int i=x;i;i-=lowbit(i)) res=max(res,c[i]);
return res;
}
void solve(int l,int r) {
if (l==r) return;
int mid=(l+r)>>1,l1=l,l2=mid+1;
for (int i=l;i<=r;i++) q[i].id<=mid ? nq[l1++]=q[i] : nq[l2++]=q[i];
for (int i=l;i<=r;i++) q[i]=nq[i];
for (int i=l,j=mid+1;i<=mid || j<=r;) {
if (j>r || (i<=mid && q[i].x<=q[j].x)) {
if (q[i].op==1) modify(q[i].y,q[i].x+q[i].y);
i++;
}
else {
if (q[j].op==2) q[j].ans=min(q[j].ans,abs(q[j].x+q[j].y-query(q[j].y)));
j++;
}
}
for (int i=l;i<=mid;i++) if (q[i].op==1) clear(q[i].y);
solve(l,mid);solve(mid+1,r);
}
int main() {
scanf("%d%d",&n,&m);X=-inf,Y=-inf;
for (int i=1;i<=n;i++) {
q[i].x=gi()+1,q[i].y=gi()+1;
X=max(q[i].x,X);Y=max(q[i].y,Y);
q[i].id=i,q[i].op=1;q[i].ans=inf;
}
for (int i=1;i<=m;i++) {
q[i+n].op=gi(),q[i+n].x=gi()+1,q[i+n].y=gi()+1;
X=max(q[i+n].x,X);Y=max(q[i+n].y,Y);
q[i+n].id=i+n;q[i+n].ans=inf;
}
n+=m;
for (int i=1;i<=n;i++) c[i]=-inf;
sort(q+1,q+1+n,cmp);solve(1,n);
for (int i=1;i<=n;i++) q[i].x=X-q[i].x+1;
sort(q+1,q+1+n,cmp);solve(1,n);
for (int i=1;i<=n;i++) q[i].y=Y-q[i].y+1;
sort(q+1,q+1+n,cmp);solve(1,n);
for (int i=1;i<=n;i++) q[i].x=X-q[i].x+1;
sort(q+1,q+1+n,cmp);solve(1,n);
for (int i=1;i<=n;i++) if (q[i].ans!=inf) printf("%d\n",q[i].ans);
return 0;
}

【bzoj2648】 SJY摆棋子的更多相关文章

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

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

  2. [BZOJ2648] SJY摆棋子 kd-tree

    2648: SJY摆棋子 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 5421  Solved: 1910[Submit][Status][Disc ...

  3. Bzoj2648 SJY摆棋子

    Time Limit: 20 Sec  Memory Limit: 128 MB Submit: 3128  Solved: 1067 Description 这天,SJY显得无聊.在家自己玩.在一个 ...

  4. BZOJ2648 SJY摆棋子(KD-Tree)

    板子题. #include<iostream> #include<cstdio> #include<cmath> #include<cstdlib> # ...

  5. 【kd-tree】bzoj2648 SJY摆棋子

    #include<cstdio> #include<cmath> #include<algorithm> using namespace std; #define ...

  6. 2019.01.14 bzoj2648: SJY摆棋子(kd-tree)

    传送门 kd−treekd-treekd−tree模板题. 题意简述:支持在平面上插入一个点,求对于一个点的最近点对. 思路:cdqcdqcdq是一种很不错的分治方法 只是好像码量有点窒息 所以我用了 ...

  7. KDTree(Bzoj2648: SJY摆棋子)

    题面 传送门 KDTree 大概就是一个分割\(k\)维空间的数据结构,二叉树 建立:每层选取一维为关键字,把中间的点拿出来,递归左右,有个\(STL\)函数nth_element可以用一下 维护:维 ...

  8. [bzoj2648]SJY摆棋子(带插入kd-tree)

    解题关键:带插入kdtree模板题. #include<iostream> #include<cstdio> #include<cstring> #include& ...

  9. luogu4169 [Violet]天使玩偶/SJY摆棋子 / bzoj2648 SJY摆棋子 k-d tree

    k-d tree + 重构的思想,就能卡过luogu和bzoj啦orz #include <algorithm> #include <iostream> #include &l ...

  10. 【BZOJ2648】SJY摆棋子(KD-Tree)

    [BZOJ2648]SJY摆棋子(KD-Tree) 题面 BZOJ Description 这天,SJY显得无聊.在家自己玩.在一个棋盘上,有N个黑色棋子.他每次要么放到棋盘上一个黑色棋子,要么放上一 ...

随机推荐

  1. 使用cmd打开java文件,报错:“错误,编码GBK的不可映射字符”

    今天使用EditPlus写了一个小程序,用cmd运行时报错--"错误,编码GBK的不可映射字符". 处理办法是用EditPlus另存为时,把编码格式由UTF-8改为ANSI. 然后 ...

  2. tomcat项目中文乱码问题解决方法

    在部署tomcat项目时经常会遇到中文乱码问题,解决的方法可参考以下步骤. 1.更改Tomcat安装目录下的conf\server.xml,指定浏览器的编码格式为"utf-8"格式 ...

  3. Web报表工具FineReport填报界面键盘操作

    对于一张填报数据较多的报表,需要用户频繁地操作鼠标.而FineReport填报界面除去按钮类型的控件,其余可以完全使用键盘而不需要用鼠标操作,对于用户而言,这将极大的节省信息录入的时间. 这里我们对填 ...

  4. Atitit ati licenseService    设计原理

    Atitit ati licenseService    设计原理 C:\0workspace\AtiPlatf\src_atibrow\com\attilax\license\LicenseX.ja ...

  5. iOS - NSMutableAttributedString富文本的实现

    NSMutableAttributedString继承于NSAttributedString(带属性的字符串)能够简单快速实现富文本的效果;不多说直接上效果图和代码,通俗易懂: (一)效果图: (二) ...

  6. Linq表达式和Lambda表达式用法对比

    什么是Linq表达式?什么是Lambda表达式?前一段时间用到这个只是,在网上也没找到比较简单明了的方法,今天就整理了一下相关知识,有空了再仔细研究研究 public Program() { List ...

  7. [Hadoop in Action] 第2章 初识Hadoop

    Hadoop的结构组成 安装Hadoop及其3种工作模式:单机.伪分布和全分布 用于监控Hadoop安装的Web工具   1.Hadoop的构造模块   (1)NameNode(名字节点)       ...

  8. Redis在游戏服务器中的应用

    排行榜游戏服务器中涉及到很多排行信息,比如玩家等级排名.金钱排名.战斗力排名等.一般情况下仅需要取排名的前N名就可以了,这时可以利用数据库的排序功能,或者自己维护一个元素数量有限的top集合.但是有时 ...

  9. android 获取网络类型名称2G 3G 4G wifi

    首先工程Manifest文件要引用: <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" ...

  10. 最实用的IT类网站及工具大集合

    1.聚合数据 大家在开发过程中,可能会用到各种各样的数据,想找一些接口来提供一些数据.比如天气预报查询,火车时刻表查询,彩票查询,身份证查询等等.有了这个接口,直接调用即可.各种各样的API接口满足你 ...