题目描述

  一个平原上有\(n\)个城市,第\(i\)个城市在点\((\cos \frac{2i\pi}{n},\sin \frac{2i\pi}{n})\)上。

  每个城市和最近的两个城市有一条直线段的路。

  此外,还有\(n-3\)条路,这些路不会和原有的路重合,这些路之间也不会相交。

  通过每条道路均要花费\(1\)的时间。

  每次给你两个城市,问你从一个城市到另一个城市最快要多久。

  \(n\leq 100000\)

题解

  先把图画出来,容易发现这是一个平面图,且这个图的对偶图是一棵树,每个点的度数不超过\(3\)。

  那么我们可以对这棵树分治(点分治边分治都可以)。

  每次选择一条边,对于每个询问,这个询问的最短路径可以经过这条边的两个端点,也可以不经过。

  那么可以从这条边的两个端点开始BFS,并更新答案。

  对于一个询问,如果这个询问的两个点不在这条边的同一侧,就可以把这个询问扔掉了,否则递归下去处理。

  时间复杂度:\(O(n\log n)\)

代码

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<ctime>
#include<cstdlib>
#include<utility>
#include<vector>
#include<queue>
using namespace std;typedef long long ll;typedef pair<int,int> pii;void open(const char *s){
#ifndef ONLINE_JUDGE
char str[100];sprintf(str,"%s.in",s);freopen(str,"r",stdin);sprintf(str,"%s.out",s);freopen(str,"w",stdout);
#endif
}int rd(){int s,c;while((c=getchar())<'0'||c>'9');s=c-'0';while((c=getchar())>='0'&&c<='9')s=s*10+c-'0';return s;}
vector<int> g[100010];
int e[100010][3];
int qu[100010][3];
int c[100010];
int ti;
int tot;
int b[100010];
queue<int> q;
int num[100010];
int ans[100010];
int d[100010];
void bfs(int x)
{
d[x]=0;
q.push(x);
ti++;
b[x]=ti;
while(!q.empty())
{
x=q.front();
q.pop();
for(auto v:g[x])
if(c[v]==tot&&b[v]!=ti)
{
d[v]=d[x]+1;
b[v]=ti;
q.push(v);
}
}
}
int cnt;
int dist(int l,int r)
{
if(r<l)
r+=cnt;
return r-l+1;
}
int belong(int l,int r,int x)
{
if(l>r)
r+=cnt;
if(x<l)
x+=cnt;
return x<=r;
}
int belong2(int l,int r,int x)
{
if(l>r)
r+=cnt;
if(x<=l)
x+=cnt;
return x<r;
}
void solve(vector<int> &id,vector<int> &qid)
{
if(!qid.size())
return;
if(id.size()<=1)
return;
tot++;
cnt=0;
for(auto v:id)
{
c[v]=tot;
num[v]=++cnt;
}
int s=0x7fffffff,x1,x2;
for(auto v1:id)
for(auto v2:g[v1])
if(c[v2]==tot)
{
int v=max(dist(num[v1],num[v2]),dist(num[v2],num[v1]));
if(v<s)
{
s=v;
x1=v1;
x2=v2;
}
}
bfs(x1);
for(auto v:qid)
ans[v]=min(ans[v],d[qu[v][1]]+d[qu[v][2]]);
bfs(x2);
for(auto v:qid)
ans[v]=min(ans[v],d[qu[v][1]]+d[qu[v][2]]);
if(id.size()<=3)
return;
vector<int> id1,id2,qid1,qid2;
for(auto v:id)
{
if(belong(num[x1],num[x2],num[v]))
id1.push_back(v);
if(belong(num[x2],num[x1],num[v]))
id2.push_back(v);
}
for(auto v:qid)
{
if(belong2(num[x1],num[x2],num[qu[v][1]])&&belong2(num[x1],num[x2],num[qu[v][2]]))
qid1.push_back(v);
if(belong2(num[x2],num[x1],num[qu[v][1]])&&belong2(num[x2],num[x1],num[qu[v][2]]))
qid2.push_back(v);
}
solve(id1,qid1);
solve(id2,qid2);
}
int n,m;
int main()
{
open("b");
scanf("%d",&n);
int x,y;
for(int i=1;i<=n-3;i++)
{
scanf("%d%d",&x,&y);
x++;
y++;
g[x].push_back(y);
g[y].push_back(x);
e[i][1]=x;
e[i][2]=y;
}
for(int i=1;i<=n;i++)
{
g[i].push_back(i%n+1);
g[i%n+1].push_back(i);
}
vector<int> id,qid;
for(int i=1;i<=n;i++)
id.push_back(i);
scanf("%d",&m);
for(int i=1;i<=m;i++)
{
scanf("%d%d",&qu[i][1],&qu[i][2]);
qu[i][1]++;
qu[i][2]++;
qid.push_back(i);
if(qu[i][1]==qu[i][2])
ans[i]=0;
else
ans[i]=0x7fffffff;
}
solve(id,qid);
for(int i=1;i<=m;i++)
printf("%d\n",ans[i]);
return 0;
}

【XSY2771】城市 分治的更多相关文章

  1. 【BZOJ2001】[HNOI2010]城市建设(CDQ分治,线段树分治)

    [BZOJ2001][HNOI2010]城市建设(CDQ分治,线段树分治) 题面 BZOJ 洛谷 题解 好神仙啊这题.原来想做一直不会做(然而YCB神仙早就切了),今天来怒写一发. 很明显这个玩意换种 ...

  2. BZOJ2001 [Hnoi2010]City 城市建设 CDQ分治

    2001: [Hnoi2010]City 城市建设 Time Limit: 20 Sec  Memory Limit: 162 MB Description PS国是一个拥有诸多城市的大国,国王Lou ...

  3. 【HNOI2010】城市建设(对时间分治 & Kruskal)

    Description \(n\) 个点 \(m\) 条边的带边权无向图.\(q\) 次操作,每次修改一条边的权值. 求每次修改后的最小生成树的边权和. Hint \(1\le n\le 2\time ...

  4. BZOJ2001 HNOI2010城市建设(线段树分治+LCT)

    一个很显然的思路是把边按时间段拆开线段树分治一下,用lct维护MST.理论上复杂度是O((M+Q)logNlogQ),实际常数爆炸T成狗.正解写不动了. #include<iostream> ...

  5. 【CDQ分治】[HNOI2010]城市建设

    题目链接 线段树分治+LCT只有80 然后就有了CDQ分治的做法 把不可能在生成树里的扔到后面 把一定在生成树里的扔到并查集里存起来 分治到l=r,修改边权,跑个kruskal就行了 由于要支持撤销, ...

  6. BZOJ2001 [Hnoi2010]City 城市建设 【CDQ分治 + kruskal】

    题目链接 BZOJ2001 题解 CDQ分治神题... 难想难写.. 比较朴素的思想是对于每个询问都求一遍\(BST\),这样做显然会爆 考虑一下时间都浪费在了什么地方 我们每次求\(BST\)实际上 ...

  7. bzoj 2001 CITY 城市建设 cdq分治

    题目传送门 题解: 对整个修改的区间进行分治.对于当前修改区间来说,我们对整幅图中将要修改的边权都先改成-inf,跑一遍最小生成树,然后对于一条树边并且他的权值不为-inf,那么这条边一定就是树边了. ...

  8. P3206 [HNOI2010]城市建设 [线段树分治+LCT维护动态MST]

    Problem 这题呢 就边权会在某一时刻变掉-众所周知LCT不支持删边的qwq- 所以考虑线段树分治- 直接码一发 如果 R+1 这个时间修改 那就当做 [L,R] 插入了一条边- 然后删的边和加的 ...

  9. BZOJ 2001: [Hnoi2010]City 城市建设

    2001: [Hnoi2010]City 城市建设 Time Limit: 20 Sec  Memory Limit: 162 MBSubmit: 1132  Solved: 555[Submit][ ...

随机推荐

  1. 模块的语法 import ,from...import....

    ------------------------积极的人在每一次忧患中都看到一个机会, 而消极的人则在每个机会都看到某种忧患 1. 认识模块 模块可以认为是一个py文件. 模块实际上是我们的py文件运 ...

  2. A4纸尺寸 web打印报告

    A4纸对应的像素尺寸: <style> @media print { .Noprn{ display:none;} .print-hidden { display: none !impor ...

  3. 培训班课程课时及费用管理系统V3.0,适合钢琴培训班、艺术培训班等

    联系QQ 564955427. ACM3.02 文件下载                    还有: 预收课时版 特点: 1. 适合主要业务是一对一课程和部分集体课培训的中小培训班(非连锁管理).考 ...

  4. 现代程序设计 homework-06

    写代码爽还是读代码爽? 当然是写代码爽好吧... 读代码明显是读+写两倍的工作量好么... 本次作业要求: 1) 把程序编译通过, 跑起来. 读懂程序,在你觉得比较难懂的地方加上一些注释,这样大家就能 ...

  5. pip ipython启动错误 Fatal error in launcher: Unable to create process using

    完整的错误提示: C:\Users\yyy>ipython3Fatal error in launcher: Unable to create process using '"c:\u ...

  6. myeclipse部署报错报空指针异常

    hib4.1+spring3+struts2项目 项目运行报错,把WEB-INF/classes目录删除后,想再重新编译并自动部署.再自动部署时总是提示错误: Errors occurred duri ...

  7. PHP--高级算法--面试

    数据结构和算法(转载) 原文地址:  https://blog.csdn.net/s1070/article/details/51174725 1.使对象可以像数组一样进行foreach循环,要求属性 ...

  8. 关于Fatal error: Paletter image not supported by webp 报错

    报错提示 Fatal error: Paletter image not supported by webp 原因是由于图片被非法编辑过(相对PHP来说)造成, 有可能是某些编辑图片的软件的格式与PH ...

  9. node-sass下载失败 关于webpack

    安装node-sass提示没有vendor目录的解决办法在node-sass目录下面新建一个vendor的空目录,然后运行npm/cnpm rebuild node-sass --save-dev即可 ...

  10. Highgo 瀚高数据库的简单搭建以及处理参数等.

    1. 获取一个瀚高数据库的安装文件 我这边只获取了 瀚高的 2.0.4 的windows x64 版本的. 来源: 同事从供应商那里获取的. 2. windows上面简单安装 很简单 exe 一路ne ...