原题

已知n个点有天使玩偶,有m次操作:

操作1:想起来某个位置有一个天使玩偶

操作2:询问离当前点最近的天使玩偶的曼哈顿距离


显然的CDQ问题,三维分别为时间,x轴,y轴。

但是这道题的问题在于最近距离怎么维护。

曼哈顿距离定义为|x2-x1|+|y2-y1|,所以把绝对值展开后一共有四种情况:

\(x2-x1+y2-y1 => x2+y2-(x1+y1)
x1-x2+y2-y1 => -x2+y2+(x1-y1)
x2-x1+y1-y2 => x2-y2+(y1-x1)
x1-x2+y1-y2 => -x2-y2+(x1+y1)\)

所以把四种情况都讨论一遍,每次处理为该情况的点对即可。

#include<cstdio>
#include<algorithm>
#include<cstring>
#define N 500010
using namespace std;
int n,m,op,ans[N],mxy,f[1000010];
struct hhh
{
int x,y,tm,op,num;
bool operator < (const hhh &b) const
{
return tm<b.tm;
}
}p[2*N],tmp[2*N]; int read()
{
int ans=0,fu=1;
char j=getchar();
for (;j<'0' || j>'9';j=getchar()) if (j=='-') fu=-1;
for (;j>='0' && j<='9';j=getchar()) ans*=10,ans+=j-'0';
return ans*fu;
} inline void add(int x,int y)
{
while (x<=mxy) f[x]=max(f[x],y),x+=x&-x;
} inline int query(int x)
{
int ans=0xc0c0c0c0;
while (x) ans=max(ans,f[x]),x-=x&-x;
return ans;
} inline void clear(int x)
{
while (x<=mxy) f[x]=0xc0c0c0c0,x+=x&-x;
} void solve(int l,int r)
{
if (l==r) return ;
int mid=(l+r)>>1;
solve(l,mid);
solve(mid+1,r);
int l1=l,l2=mid+1;
for (int i=l;i<=r;i++)
{
if (l1<=mid && (l2>r || p[l1].x<p[l2].x)) tmp[i]=p[l1++];
else tmp[i]=p[l2++];
} for (int i=l;i<=r;i++)
{
p[i]=tmp[i];
if (p[i].tm>mid && p[i].op) ans[p[i].num]=min(ans[p[i].num],p[i].x+p[i].y-query(p[i].y));
if (p[i].tm<=mid && !p[i].op) add(p[i].y,p[i].x+p[i].y);
}
for (int i=l;i<=r;i++)
{
if (p[i].tm<=mid && !p[i].op)
clear(p[i].y);
} for (int i=l;i<=r;i++)
{
if (p[i].tm>mid && p[i].op) ans[p[i].num]=min(ans[p[i].num],p[i].x-p[i].y-query(mxy-p[i].y));
if (p[i].tm<=mid && !p[i].op) add(mxy-p[i].y,p[i].x-p[i].y);
}
for (int i=l;i<=r;i++)
{
if (p[i].tm<=mid && !p[i].op)
clear(mxy-p[i].y);
} for (int i=r;i>=l;i--)
{
if (p[i].tm>mid && p[i].op) ans[p[i].num]=min(ans[p[i].num],-p[i].x+p[i].y-query(p[i].y));
if (p[i].tm<=mid && !p[i].op) add(p[i].y,p[i].y-p[i].x);
}
for (int i=r;i>=l;i--)
{
if (p[i].tm<=mid && !p[i].op)
clear(p[i].y);
} for (int i=r;i>=l;i--)
{
if (p[i].tm>mid && p[i].op) ans[p[i].num]=min(ans[p[i].num],-p[i].x-p[i].y-query(mxy-p[i].y));
if (p[i].tm<=mid && !p[i].op) add(mxy-p[i].y,-p[i].x-p[i].y);
}
for (int i=r;i>=l;i--)
{
if (p[i].tm<=mid && !p[i].op)
clear(mxy-p[i].y);
}
} int main()
{
freopen("1.in","r",stdin);
freopen("1.out","w",stdout);
n=read();
m=read();
for (int i=1;i<=n;i++)
{
p[i].x=read()+1;
p[i].y=read()+1;
p[i].tm=i;
mxy=max(mxy,p[i].y);
}
for (int i=1;i<=m;i++)
{
op=read();
p[n+i].x=read()+1;
p[n+i].y=read()+1;
p[n+i].tm=n+i;
mxy=max(mxy,p[n+i].y);
if (op==2)
p[n+i].num=++ans[0],p[n+i].op=1;
}
for (int i=1;i<=ans[0];i++) ans[i]=0x3f3f3f3f;
mxy++;
for (int i=0;i<=mxy;i++) f[i]=0xc0c0c0c0;
sort(p+1,p+n+m+1);
solve(1,n+m);
for (int i=1;i<=ans[0];i++)
printf("%d\n",ans[i]);
return 0;
}

[bzoj] 2716 天使玩偶 || CDQ分治的更多相关文章

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

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

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

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

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

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=2716 怎么KD树跑得都那么快啊..我写的CDQ分治被暴虐 做四遍CDQ分治,每次求一个 ...

  4. bzoj 2716 天使玩偶(CDQ分治,BIT)

    [题目链接] http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=29234 [题意] 询问当前点与已知点的最小曼哈顿距离. [思路 ...

  5. BZOJ 2716 Violet 3 天使玩偶 CDQ分治

    题目大意:初始给定平面上的一个点集.提供两种操作: 1.将一个点增加点集 2.查询距离一个点最小的曼哈顿距离 K-D树是啥...不会写... 我仅仅会CDQ分治 对于一个询问,查询的点与这个点的位置关 ...

  6. BZOJ.2716.[Violet3]天使玩偶(CDQ分治 坐标变换)

    题目链接 考虑对于两个点a,b,距离为|x[a]-x[b]|+|y[a]-y[b]|,如果a在b的右上,那我们可以把绝对值去掉,即x[a]+y[a]-(x[b]+y[b]). 即我们要求满足x[b]& ...

  7. bzoj 2716 天使玩偶 —— K-D树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2716 果然和 bzoj 2648 是一样的吧: 只是数组要迷之开大,3e5+5 会RE? 代 ...

  8. 洛谷P4169 天使玩偶 CDQ分治

    还是照着CDQ的思路来. 但是有一些改动: 要求4个方向的,但是可爱的CDQ分治只能求在自己一个角落方向上的.怎么办?旋转!做4次就好了. 统计的不是和,而是——max!理由如下: 设当前点是(x,y ...

  9. CH 4701 - 天使玩偶 - [CDQ分治]

    题目链接:传送门 关于CDQ分治(参考李煜东<算法竞赛进阶指南>): 对于一系列操作,其中的任何一个询问操作,其结果必然等价于:初始值 + 此前所有的修改操作产生的影响. 假设共有 $m$ ...

随机推荐

  1. Java分享笔记:Map集合(接口)的基本方法程序演示

    package pack02; import java.util.*; public class MapDemo { public static void main(String[] args) { ...

  2. lincode 680 Split String

    Split String    描述 笔记 数据 评测 Give a string, you can choose to split the string after one character or ...

  3. 屏蔽datatable错误提示

    $.fn.dataTable.ext.errMode = 'none'; //不显示任何错误信息// 以下为发生错误时的事件处理,如不处理,可不管.$('#productionRequestItems ...

  4. RPC框架基础概念理解以及使用初体验

    RPC:Remote Procedure Call(远程服务调用) RPC是做什么的 通过RPC框架机器A某个进程可以通过网络调用机器B上的进程方法,就像在本地上调用一样. RPC可以基于HTTP或者 ...

  5. LeetCode977. 有序数组的平方

    问题:977. 有序数组的平方 给定一个按非递减顺序排序的整数数组 A,返回每个数字的平方组成的新数组,要求也按非递减顺序排序. 示例 1: 输入:[-4,-1,0,3,10] 输出:[0,1,9,1 ...

  6. 在React Native中集成热更新

    最近,在项目DYTT集成了热更新,简单来说,就是不用重新下载安装包即可达到更新应用的目的,也不算教程吧,这里记录一下. 1.热更新方案 目前网上大概有两个比较广泛的方式,分别是 react-nativ ...

  7. 柱状图多系列php动态实现(ec)

    <?php require_once 'data.php'; $arr1=$a->sum('answer','ask_id=1'); $arr2=$a->sum('answer',' ...

  8. vim+软件安装——06

    vim在命令模式下的操作: 1.上下左右键可以自由走动 2.l 键 光标向右移动一个位置 3.h键 光标向左移动一个位置 4.j键 光标向下移动一行 5.k键 光标向上移动一行 6.^键 光标移动到当 ...

  9. 猜数字问题 python

    猜数字问题,要求如下: ① 随机生成一个整数 ② 猜一个数字并输入 ③ 判断是大是小,直到猜正确 ④ 判断时间提示:需要用time模块.random模块该题目不需要创建函数 import random ...

  10. Eclipse字体修改

    第一步: 第二步: 第三步: 第四步: 第五步: 第六步: