Description

给出一张四连通网格图,其中有 \(n\) 个点是连通的,维护以下两种操作:

1.把某个点变黑

2.给出一个白点,查询离这个白点最近的黑点的距离

题面

Solution

我们把每一行看作一个节点,建立一棵树

然后点分治维护这两个操作即可

实际上就是动态加入黑点,查询离某个点最近的黑点

唯一的区别在于:

走到这个节点之后还会在这个节点所代表的行里面走,设查询的为 \(x\),某个黑点为 \(u\)

要求: \(min(dis[u]+dis[x]+|y_u-y_x|)\)

\(dis[i]\) 表示从 \(i\) 出发走到这个重心所代表的行的最短路,\(y_i\) 表示走最短路碰到这个节点所代表的行的纵坐标

两者都可以从重心 \(bfs\) 求出

然后对于每一个重心维护数状数组查询 \(min(dis[u]+dis[x]+|y_u-y_x|)\)

把绝对值拆掉,用两个树状数组维护就行了,分别对应前后缀查询

值得注意的是:

这题的重心要是带权重心,否则预处理就会 \(TLE\)

#include<bits/stdc++.h>
#define pb push_back
using namespace std;
const int N=3e5+10,M=22,inf=1e9;
int n,m=0,tt=0,cnt=0,b[N],s[N],Lx[N],Ly[N],fa[N];
vector<int>S[N],G[N],E[N];
map<int,int>id[N];
inline void priwork(){
for(int i=1;i<=m;i++){
if(S[i].empty())continue;
sort(S[i].begin(),S[i].end());
for(int j=0,si=S[i].size(),x,k=0;j<si;j++){
x=S[i][j];id[i][x]=++tt;
if(j && S[i][j-1]+1==x)G[tt].pb(tt-1),G[tt-1].pb(tt);
else ++cnt,Lx[cnt]=i,Ly[cnt]=j;
b[tt]=cnt;s[cnt]++;
while(k<(int)S[i-1].size() && S[i-1][k]<x)k++;
if(k<(int)S[i-1].size() && S[i-1][k]==x){
E[b[id[i-1][x]]].pb(cnt),E[cnt].pb(b[id[i-1][x]]);
G[id[i-1][x]].pb(tt),G[tt].pb(id[i-1][x]);
}
}
}
for(int i=1;i<=n;i++){
sort(E[i].begin(),E[i].end());
E[i].erase(unique(E[i].begin(),E[i].end()),E[i].end());
}
}
int son[N]={inf},sz[N],rt=0,sum,p[N][M],dis[N][M];bool vis[N];
inline void getroot(int x,int last){
sz[x]=s[x];son[x]=0;
for(int i=E[x].size()-1;i>=0;i--){
int u=E[x][i];
if(u==last || vis[u])continue;
getroot(u,x);sz[x]+=sz[u];son[x]=max(son[x],sz[u]);
}
son[x]=max(son[x],sum-sz[x]);
if(son[x]<son[rt])rt=x;
}
vector<int>t1[N],t2[N];int dep[N];
inline void add1(int o,int x,int t){
for(int i=x;i<=s[o];i+=(i&(-i)))t1[o][i]=min(t1[o][i],t);
}
inline int qry1(int o,int x){
int ret=inf;
for(int i=x;i>=1;i-=(i&(-i)))ret=min(ret,t1[o][i]);
return ret;
}
inline void add2(int o,int x,int t){
for(int i=x;i>=1;i-=(i&(-i)))t2[o][i]=min(t2[o][i],t);
}
inline int qry2(int o,int x){
int ret=inf;
for(int i=x;i<=s[o];i+=(i&(-i)))ret=min(ret,t2[o][i]);
return ret;
}
inline void build(int x,int d){
queue<int>Q;
for(int i=Ly[x],u;i<Ly[x]+s[x];i++){
u=id[Lx[x]][S[Lx[x]][i]];
p[u][d]=i-Ly[x]+1;dis[u][d]=0;Q.push(u);
}
while(!Q.empty()){
int x=Q.front();Q.pop();
for(int i=G[x].size()-1;i>=0;i--){
int u=G[x][i];
if(vis[b[u]] || p[u][d])continue;
p[u][d]=p[x][d],dis[u][d]=dis[x][d]+1,Q.push(u);
}
}
for(int i=0;i<=s[x];i++)t1[x].pb(inf),t2[x].pb(inf);
}
inline void solve(int x,int d){
vis[x]=1;dep[x]=d;build(x,d);
for(int i=E[x].size()-1;i>=0;i--){
int u=E[x][i];if(vis[u])continue;
rt=0;sum=sz[u];getroot(u,x);
fa[rt]=x;solve(rt,d+1);
}
}
inline void Modify(int y){
int x=b[y],d;
while(x){
d=dep[x];
add1(x,p[y][d],dis[y][d]-p[y][d]);
add2(x,p[y][d],dis[y][d]+p[y][d]);
x=fa[x];
}
}
inline int query(int y){
int ret=inf,d,x=b[y];
while(x){
d=dep[x];
ret=min(ret,dis[y][d]+p[y][d]+qry1(x,p[y][d]));
ret=min(ret,dis[y][d]-p[y][d]+qry2(x,p[y][d]));
x=fa[x];
}
if(ret<N)return ret;
return -1;
}
int main(){
int x,y,Q,op;
scanf("%d",&n);
for(int i=1;i<=n;i++){
scanf("%d%d",&x,&y);
m=max(x,m);S[x].pb(y);
}
priwork();
rt=0;sum=n;getroot(1,1);solve(1,0);
cin>>Q;
while(Q--){
scanf("%d%d%d",&op,&x,&y);
if(op==2)printf("%d\n",query(id[x][y]));
else Modify(id[x][y]);
}
return 0;
}

Codeforces 936E. Iqea的更多相关文章

  1. python爬虫学习(5) —— 扒一下codeforces题面

    上一次我们拿学校的URP做了个小小的demo.... 其实我们还可以把每个学生的证件照爬下来做成一个证件照校花校草评比 另外也可以写一个物理实验自动选课... 但是出于多种原因,,还是绕开这些敏感话题 ...

  2. 【Codeforces 738D】Sea Battle(贪心)

    http://codeforces.com/contest/738/problem/D Galya is playing one-dimensional Sea Battle on a 1 × n g ...

  3. 【Codeforces 738C】Road to Cinema

    http://codeforces.com/contest/738/problem/C Vasya is currently at a car rental service, and he wants ...

  4. 【Codeforces 738A】Interview with Oleg

    http://codeforces.com/contest/738/problem/A Polycarp has interviewed Oleg and has written the interv ...

  5. CodeForces - 662A Gambling Nim

    http://codeforces.com/problemset/problem/662/A 题目大意: 给定n(n <= 500000)张卡片,每张卡片的两个面都写有数字,每个面都有0.5的概 ...

  6. CodeForces - 274B Zero Tree

    http://codeforces.com/problemset/problem/274/B 题目大意: 给定你一颗树,每个点上有权值. 现在你每次取出这颗树的一颗子树(即点集和边集均是原图的子集的连 ...

  7. CodeForces - 261B Maxim and Restaurant

    http://codeforces.com/problemset/problem/261/B 题目大意:给定n个数a1-an(n<=50,ai<=50),随机打乱后,记Si=a1+a2+a ...

  8. CodeForces - 696B Puzzles

    http://codeforces.com/problemset/problem/696/B 题目大意: 这是一颗有n个点的树,你从根开始游走,每当你第一次到达一个点时,把这个点的权记为(你已经到过不 ...

  9. CodeForces - 148D Bag of mice

    http://codeforces.com/problemset/problem/148/D 题目大意: 原来袋子里有w只白鼠和b只黑鼠 龙和王妃轮流从袋子里抓老鼠.谁先抓到白色老鼠谁就赢. 王妃每次 ...

随机推荐

  1. 系统禁用执行FIPS政策导致程序发生“调用的目标发生了异常”

    工具是使用AES-256-CBC加密算法 问题 最近有客户反映, 在使用我们工具时候,会出现“调用的目标发生了异常”错误, 接到反馈之后, 我们进行了很多测试,甚至得到客户系统信息和framework ...

  2. 两款Mongodb可视化工具

    在某些场景下,相比传统的关系数据库和NoSQL数据库,Mongodb拥有不可替代的优势. 例如,最近我需要为收集的大量网站进行分类.实际情况是,一个网站可能同时有多个标签,想象一下新浪网,它既是门户站 ...

  3. 【06】循序渐进学 docker:跨主机通信

    写在前面的话 目前解决容器跨主机通信的方案有很多种,这里给出的只是其中的一种,而且还不是最好的方案,不过归根结底,大同小异.在学习 docker swarm 之前,大家可以先看看这种. 啥是 over ...

  4. 原生态js单个点击展开收缩

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  5. [译文]casperjs使用说明-使用命令行

    使用命令行 Casperjs使用内置的phantomjs命令行解析器,在cli模块里,它传递参数位置的命名选项 但是不要担心不能熟练操控CLI模块的API,一个casper实例已经包含了cli属性,允 ...

  6. Python3之 contextlib

    Python中当我们们打开文本时,通常会是用with语句,with语句允许我们非常方便的使用资源,而不必担心资源没有关闭. with open('/path/filename', 'r') as f: ...

  7. c语言-折半查找的函数

    void search(int n,int num[],char name[N][10]) { int top,bottom,middle,location,flag; top=0; bottom=N ...

  8. 北航软院2015级C#期末考试部分考题讲解

    洗洗睡了吧,我怎么知道明天的考试题目! 或者 你明年补考可以过来看看:) 晚安.

  9. PHP错误与异常处理

    https://www.cnblogs.com/zyf-zhaoyafei/p/6928149.html 请一定要注意,没有特殊说明:本例 PHP Version < 7 说起PHP异常处理,大 ...

  10. DOM操作 045

    一 什么是DOM DOM : 文档对象模型 它为文档提供了结构化表示 并定义了如何通过脚本来访问文档结构 . 目的就是为了能让js操作HTML元素而制定的一个规范 . DOM树(一切都是节点): 元素 ...