BZOJ 2716/2648 SJY摆棋子 (三维偏序CDQ+树状数组)
题目大意:
这明明是一道KD-Tree,CDQ分治是TLE的做法
化简式子,$|x1-x2|-|y1-y2|=(x1+y1)-(x2+y2)$
而$CDQ$分治只能解决$x1 \leq x2,y1 \leq y2$的情况
把每次插入操作都相当于一个三元组$<x,y,t>$,权值是$x+y$。这就是一个三维偏序问题,用树状数组维护最大值即可
所以通过坐标变换,跑$4$次$CDQ$就行了?
没错,你会像我一样T得飞起
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N1 1001000
#define M1 1001000
#define ll long long
#define dd double
#define inf 233333333
using namespace std; int gint()
{
int ret=,fh=;char c=getchar();
while(c<''||c>''){if(c=='-')fh=-;c=getchar();}
while(c>=''&&c<=''){ret=ret*+c-'';c=getchar();}
return ret*fh;
}
int n,m,ma,nm,mx,my;
struct BIT{
int ma[M1];
void update(int x,int w){ for(int i=x;i<=my;i+=(i&(-i))) ma[i]=max(ma[i],w); }
int query(int x){ int ans=-inf; for(int i=x;i;i-=(i&(-i))) ans=max(ans,ma[i]); return ans; }
void clr(int x){ for(int i=x;i<=my;i+=(i&(-i))) ma[i]=-inf;}
void init(){ for(int i=;i<=my;i++) ma[i]=-inf; }
}s;
struct node{int x,y,p,t,ans;}a[N1],tmp[N1];
int cmp1(node s1,node s2){ if(s1.x!=s2.x) return s1.x<s2.x; if(s1.y!=s2.y) return s1.y<s2.y; return s1.p<s2.p;}
int cmpt(node s1,node s2){ return s1.t<s2.t; }
int que[N1],tl; void CDQ(int L,int R)
{
if(R-L<=) return;
int i,j,k,M=(L+R)>>;
for(i=L,j=M,k=L;k<R;k++)
{
if(a[k].t<M) tmp[i++]=a[k];
else tmp[j++]=a[k];
}
for(k=L;k<R;k++) a[k]=tmp[k];
CDQ(L,M);
for(i=L,j=M;i<M&&j<R;)
{
if(cmp1(a[i],a[j])){ if(a[i].p==) { s.update(a[i].y,a[i].x+a[i].y); que[++tl]=i; } i++; }
else{ if(a[j].p==) a[j].ans=min(a[j].ans,(a[j].x+a[j].y)-s.query(a[j].y)); j++; }
}
while(j<R){ if(a[j].p==) a[j].ans=min(a[j].ans,(a[j].x+a[j].y)-s.query(a[j].y)); j++; }
while(tl){ s.clr(a[que[tl--]].y); }
CDQ(M,R);
for(i=L,j=M,k=;i<M&&j<R;)
{
if(cmp1(a[i],a[j])) tmp[++k]=a[i++];
else tmp[++k]=a[j++];
}
while(i<M){ tmp[++k]=a[i++]; }
while(j<R){ tmp[++k]=a[j++]; }
for(k=L;k<R;k++) a[k]=tmp[k-L+];
}
int p[N1]; int main()
{
int i,j; n=gint(); m=gint(); nm=n+m;
for(i=;i<=n;i++){ a[i].p=; a[i].x=gint()+; a[i].y=gint()+; a[i].t=i; a[i].ans=inf; mx=max(mx,a[i].x); my=max(my,a[i].y); }
for(i=n+;i<=n+m;i++){ a[i].p=gint(); a[i].x=gint()+; a[i].y=gint()+; a[i].t=i; a[i].ans=inf; mx=max(mx,a[i].x); my=max(my,a[i].y); }
sort(a+,a+nm+,cmp1); s.init();
CDQ(,nm+);
for(i=;i<=nm;i++){ a[i].x=mx-a[i].x+; } sort(a+,a+nm+,cmp1);
CDQ(,nm+);
for(i=;i<=nm;i++){ a[i].y=my-a[i].y+; } sort(a+,a+nm+,cmp1);
CDQ(,nm+);
for(i=;i<=nm;i++){ a[i].x=mx-a[i].x+; } sort(a+,a+nm+,cmp1);
CDQ(,nm+);
sort(a+,a+nm+,cmpt);
for(i=n+;i<=nm;i++) if(a[i].p==) printf("%d\n",a[i].ans);
return ;
}
BZOJ 2716/2648 SJY摆棋子 (三维偏序CDQ+树状数组)的更多相关文章
- BZOJ 3295 [CQOI2011]动态逆序对 (三维偏序CDQ+树状数组)
题目大意: 题面传送门 还是一道三维偏序题 每次操作都可以看成这样一个三元组 $<x,w,t>$ ,操作的位置,权值,修改时间 一开始的序列看成n次插入操作 我们先求出不删除时的逆序对总数 ...
- BZOJ 1176/2683 Mokia (三维偏序CDQ+树状数组)
题目大意: 洛谷传送门 三维偏序裸题.. 每次操作都看成一个三元组$<x,y,t>$,表示$x,y$坐标和操作时间$t $ 询问操作拆成$4$个容斥 接下来就是$CDQ$了,外层按t排序, ...
- BZOJ 2141 排队 (三维偏序CDQ+树状数组)
题目大意:略 洛谷传送门 和 [CQOI2015]动态逆序对 这道题一样的思路 一开始的序列视为$n$次插入操作 把每次交换操作看成四次操作,删除$x$,删除$y$,加入$x$,加入$y$ 把每次操作 ...
- BZOJ 3262 陌上花开 (三维偏序CDQ+树状数组)
题目大意: 题面传送门 三维偏序裸题 首先,把三元组关于$a_{i}$排序 然后开始$CDQ$分治,回溯后按$b_{i}$排序 现在要处理左侧对右侧的影响了,显然现在左侧三元组的$a_{i}$都小于等 ...
- 洛谷P3810-陌上开花(三维偏序, CDQ, 树状数组)
链接: https://www.luogu.org/problem/P3810#submit 题意: 一个元素三个属性, x, y, z, 给定求f(b) = {ax <= bx, ay < ...
- 【bzoj 2716】[Violet 3]天使玩偶 (CDQ+树状数组)
题目描述 Ayu 在七年前曾经收到过一个天使玩偶,当时她把它当作时间囊埋在了地下.而七年后 的今天,Ayu 却忘了她把天使玩偶埋在了哪里,所以她决定仅凭一点模糊的记忆来寻找它. 我们把 Ayu 生活的 ...
- bzoj3262陌上花开 三维数点 cdq+树状数组
大早上的做了一道三维数点一道五位数点,神清气爽! 先给一维排序,变成一个奇怪的动态的二维数点(相当于有一个扫描面扫过去,导致一系列的加点和询问) 然后cdq分治,再变回静态,考虑前半段对后半段的影响 ...
- 【BZOJ】2648: SJY摆棋子 & 2716: [Violet 3]天使玩偶(kdtree)
http://www.lydsy.com/JudgeOnline/problem.php?id=2716 http://www.lydsy.com/JudgeOnline/problem.php?id ...
- bzoj 2648: SJY摆棋子&&2716: [Violet 3]天使玩偶 --kdtree
2648: SJY摆棋子&&2716: [Violet 3]天使玩偶 Time Limit: 20 Sec Memory Limit: 128 MB Description 这天,S ...
随机推荐
- JAVA版本号微信公众账号开源项目版本号公布-jeewx1.0(捷微)
JeeWx, 敏捷微信开发,简称"捷微". 捷微是一款免费开源的微信公众账号开发平台. 平台介绍: 一.简单介绍 jeewx是一个开源,高效.敏捷的微信开发平台採用JAVA语言,它 ...
- C#调用C++回调函数的问题
C++的回调函数中有一个参数是,是返回一个字符串,原则如下: typedef void (*TDataEvent)(char *AData ,int ALen); 其中char ...
- POJ2576 Tug of War 二维背包
题目大意 一群人拔河,给出每个人的重量,要求两队人数之差不超过1人,且每队总重量之差最小. 思路 选出严格总人数一半(或+1)的人为一队,在该队重量不超过所有人总重量一半的情况下,使其重量最大. 人数 ...
- JFinal Starting scanner at interval of 5 seconds.报错
Starting JFinal 2.0 Starting scanner at interval of 5 seconds. Starting web server on port: 80 Excep ...
- 圆角矩形“RoundRectShape”使用详解
圆角矩形 常用作一些组件的背景 构造函数: RoundRectShape(float[] outerRadii, RectF inset, float[] innerRadii) Specifies ...
- Linux - 控制台界面,虚拟界面,字符界面
tty控制台终端. pts虚拟终端. tty1 图形界面. tty2 字符界面. Ctrl+Alt+F2-6 在字符界面下,通过Alt+F2 切换回来.或者切换到其他的字符界面. Alt+F2 pts ...
- 深度学习将会变革NLP中的中文分词——TODO 待好好细看
见:https://www.leiphone.com/news/201608/IWvc75oJglAIsDvJ.html TODO 待好好细看
- 【转】不要使用SBJSON(json-framework)
原文网址:http://blog.devtang.com/2012/05/05/do-not-use-sbjson/ 不知道为什么,在iOS开发中,有很多人使用 SBJSON (又被称作json-fr ...
- linux编译安装gdb7.10.1
1.下载GDB7.10.1安装包 #wget http://ftp.gnu.org/gnu/gdb/gdb-7.10.1.tar.gz 2.解压 #.tar.gz 3.创建安装目录 #/ #cd /u ...
- C++批量加载动态库函数方法
1.枚举定义enum { // 0 - GigE DLL (implicitly called) Func_isVersionCompliantDLL, Func_isDriver ...