P4169 [Violet]天使玩偶/SJY摆棋子

CDQ分治的题目.

我们发现题目要我们求的\(|A_x-B_x|+|A_y-B_y|\)的绝对值号比较恶心.

试想一下怎么去掉

如果所有的点都在我们当前求的点的左下方(就是只考虑在他坐下方的点对他的贡献). 我们怎么求?

那么就要我们求\(min{A_x-B_x+A_y-B_y}\)(假设\(A\)为询问的点)

那么其实就是让我们求\(A_x+A_y-max(B_x+B_y)\)

也因为要满足左下角的限制

其实就是满足

\(B_x<=A_x\)同时满足\(B_y<=A_y\)中最大的\(B_x+B_y\)

这个是可以直接CDQ的

但是我们只考虑了左下方的点对他的贡献,很明显,这是不够的

那么我们就想办法依次将左上,右上,右下全部转化为左下

就是通过同最大值域的加减来改变他们的左边但不改变相对关系

注意常数就好了

另外,这种情况下归并排序的时间复杂度比快排优秀太多了

#include<bits/stdc++.h>
using namespace std;
const int N = 6e5 + 3;
const int M = 1e6 + 3;
const int INF = 2e9;
int n,m;
int max_x,max_y;
int ans[N];
struct Q{
int type;
int id;
int xi;
int yi;
int ans;
}q[N],p[N],h[N];
Q g[N];
struct BIT{
int c[M];
inline void ins(int x,int v){
for(;x <= max_y;x += x & -x) c[x] = max(c[x],v);
}
inline int query(int x){
int res = 0;
for(;x;x -= x & -x) res = max(res,c[x]);
return res;
}
inline void clear(int x){
for(;x <= max_y;x += x & -x) c[x] = 0;
}
}T;
int xx[N],yy[N];
inline bool cmp1(Q x,Q y){
return x.id < y.id;
}
inline bool cmp2(Q x,Q y){
if(x.xi != y.xi)
return x.xi < y.xi;
return x.yi < y.yi;
}
inline int read(){
int x = 0;int flag = 0;
char ch = getchar();
while(!isdigit(ch)){
if(ch == '-') flag = 1;
ch = getchar();
}
while(isdigit(ch)){
x = (x<<1) + (x<<3) + (ch^'0');
ch = getchar();
}
if(flag) x = -x;
return x;
}
inline void solve(int l,int r){
if(l == r) return;
int mid = (l + r) >> 1;
solve(l,mid);solve(mid + 1,r);
//sort(p + l,p + mid + 1,cmp2);
//sort(p + mid + 1,p + r + 1,cmp2);
// nj printf("%d %d\n",l,r);
int now = l;
int ll = l,rr = mid + 1,nn = l - 1; // cout << 1 << endl;
for(int i = mid + 1;i <= r;++i){
while(i <= r && p[i].type != 2) ++i;
if(i > r) break;
for(;now <= mid && p[now].xi <= p[i].xi;++now) if(p[now].type == 1) T.ins(p[now].yi,p[now].xi + p[now].yi);
int t = T.query(p[i].yi);
if(t) ans[p[i].id] = min(ans[p[i].id],p[i].xi + p[i].yi - t);
}
for(int i = l;i < now;++i) if(p[i].type == 1) T.clear(p[i].yi);
while(ll <= mid && rr <= r){
if(p[ll].xi <= p[rr].xi) h[++nn] = p[ll++];
else h[++nn] = p[rr++];
}
while(ll <= mid) h[++nn] = p[ll++];
while(rr <= r) h[++nn] = p[rr++];
for(int i = l;i <= r;++i) p[i] = h[i];
}
inline void del(){
int rx = 0,ry = 0;m = 0;
for(int i = 1;i <= n;++i)
if(p[i].type == 1) rx = max(p[i].xi,rx),ry = max(p[i].yi,ry);
for(int i = 1;i <= n;++i){
if((p[i].xi <= rx && p[i].yi <= ry) || p[i].type == 2) g[++m] = p[i];
}
for(int i = 1;i <= m;++i) p[i] = g[i];
}
int main(){
n = read(),m = read();
for(int i = 1;i <= n;++i){
q[i].xi = read() + 1,q[i].yi = read() + 1;
max_x = max(max_x,q[i].xi),max_y = max(max_y,q[i].yi);
q[i].id = i;
q[i].type = 1;
}
for(int i = 1;i <= m;++i){
q[n + i].type = read();q[n + i].xi = read() + 1;q[n + i].yi = read() + 1;
q[n + i].id = i + n;
max_x = max(max_x,q[n + i].xi),max_y = max(max_y,q[n + i].yi);
}
n += m;
for(int i = 1;i <= n;++i) p[i] = q[i];
for(int i = 1;i <= n;++i) ans[i] = INF;
max_y = max(max_x,max_y) + 1;
solve(1,n);
for(int i = 1;i <= n;++i) p[i] = q[i],p[i].xi = max_y - p[i].xi;
solve(1,n);
for(int i = 1;i <= n;++i) p[i] = q[i],p[i].yi = max_y - p[i].yi;
solve(1,n);
for(int i = 1;i <= n;++i) p[i] = q[i],p[i].xi = max_y - p[i].xi,p[i].yi = max_y - p[i].yi;
solve(1,n);
for(int i = 1;i <= n;++i) if(ans[i] != INF) printf("%d\n",ans[i]);
return 0;
}

天使玩偶/SJY摆棋子的更多相关文章

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

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

  2. 【LG4169】[Violet]天使玩偶/SJY摆棋子

    [LG4169][Violet]天使玩偶/SJY摆棋子 题面 洛谷 题解 至于\(cdq\)分治的解法,以前写过 \(kdTree\)的解法好像还\(sb\)一些 就是记一下子树的横.纵坐标最值然后求 ...

  3. 洛谷 P4169 [Violet]天使玩偶/SJY摆棋子 解题报告

    P4169 [Violet]天使玩偶/SJY摆棋子 题目描述 \(Ayu\)在七年前曾经收到过一个天使玩偶,当时她把它当作时间囊埋在了地下.而七年后 的今天,\(Ayu\) 却忘了她把天使玩偶埋在了哪 ...

  4. luoguP4169 [Violet]天使玩偶/SJY摆棋子 K-Dtree

    P4169 [Violet]天使玩偶/SJY摆棋子 链接 luogu 思路 luogu以前用CDQ一直过不去. bzoj还是卡时过去的. 今天终于用k-dtree给过了. 代码 #include &l ...

  5. 洛谷P4169 [Violet]天使玩偶/SJY摆棋子(CDQ分治)

    [Violet]天使玩偶/SJY摆棋子 题目传送门 解题思路 用CDQ分治开了氧气跑过. 将输入给的顺序作为第一维的时间,x为第二维,y为第三维.对于距离一个询问(ax,ay),将询问分为四块,左上, ...

  6. [Violet]天使玩偶/SJY摆棋子 [cdq分治]

    P4169 [Violet]天使玩偶/SJY摆棋子 求离 \((x,y)\) 最近点的距离 距离的定义是 \(|x1-x2|+|y1-y2|\) 直接cdq 4次 考虑左上右上左下右下就可以了-略微卡 ...

  7. P4169 [Violet]天使玩偶/SJY摆棋子

    题目背景 感谢@浮尘ii 提供的一组hack数据 题目描述 Ayu 在七年前曾经收到过一个天使玩偶,当时她把它当作时间囊埋在了地下.而七年后 的今天,Ayu 却忘了她把天使玩偶埋在了哪里,所以她决定仅 ...

  8. LG4169 [Violet]天使玩偶/SJY摆棋子

    题意 Ayu 在七年前曾经收到过一个天使玩偶,当时她把它当作时间囊埋在了地下.而七年后 的今天,Ayu 却忘了她把天使玩偶埋在了哪里,所以她决定仅凭一点模糊的记忆来寻找它. 我们把 Ayu 生活的小镇 ...

  9. 洛谷P4169 [Violet]天使玩偶/SJY摆棋子

    %%%神仙\(SJY\) 题目大意: 一个二维平面,有两种操作: \(1.\)增加一个点\((x,y)\) \(2.\)询问距离\((x,y)\)曼哈顿最近的一个点有多远 \(n,m\le 300 0 ...

随机推荐

  1. Mysql里查询字段为Json格式的数据模糊查询以及分页方法

    public void datagrid(CustomFormEntity customForm,HttpServletRequest request, HttpServletResponse res ...

  2. 【C++】位运算实现加减乘除

    #include<iostream> #include<assert.h> using namespace std; // 位运算实现加减乘除 int myAdd(int nu ...

  3. php 正则学习取反符号~

    php 正则学习取反符号~ ~(<a .*?>.*?</a>|<.*?>)~i 先看正则图形,有点偏差,但可以初步看出结果. 关于 ~ 是取反符号,看下面说明.

  4. 字体图标font-awesome

    其实有一些常见的图标使用字体图标比使用img来得好 Font Awesome 官网:http://fortawesome.github.io/Font-Awesome/ 字体代码:http://for ...

  5. SVN过滤设置 标签: svn 2015-07-29 17:39 953人阅读 评论(35) 收藏

    为了方便管理我们的系统版本,很多人会用到SVN,开发中我们经常用到SVN插件, 但是对于某些文件的缓存来说, 我们只要有操作缓存便会保存一次, 每次提交很是麻烦, 可能有的文件或者文件夹我们并不想提交 ...

  6. vue事件获取事件对象,vue获取事件源,vue event.currentTarget

    js的事件,如点击事件,可以直接用this获取事件对象,而jQuery可以使用$(this)来获取事件对象.vue必须借助事件的 event 对象 的 currentTarget 才能获取事件对象 v ...

  7. python3中的zip函数

    zip函数的作用: zip函数接受任意多个可迭代对象作为参数,将对象中对应的元素打包成一个tuple,然后返回一个可迭代的zip对象. 这个可迭代对象可以使用循环的方式列出其元素 若多个可迭代对象的长 ...

  8. PDM->OOM->C#实体类生成时,对Blob类型字段的处理

    pdm中的Blob字段生成OOM时,自动变成了string类型,再生成实体类时也是string 如何将oom中对应的blob字段设置为Byte[]类型,目前没找到方法, 只能通过脚本,将生成后的OOM ...

  9. oracle trunc(d1[,c1])

    [功能]:返回日期d1所在期间(参数c1)的第一天日期 [参数]:d1日期型,c1为字符型(参数),c1默认为j(即当前日期) [参数表]:c1对应的参数表: 最近0点日期: 取消参数c1或j 最近的 ...

  10. 是readdir,还是readdir_r

    readdir的原型如下: struct dirent *readdir(DIR *dirp); 因为内部使用了静态数据,所以readdir被认为不是线程安全的函数,POSIX[i]标准这样描述: T ...