UVA1464 Traffic Real Time Query System
传送门:https://www.luogu.com.cn/problem/UVA1464
看到这道题,求必经的点数,还是无向图。那么妥妥的圆方树。圆方树上的任意两圆点间的路径必定是圆点方点相交错的,对于树上的两点来说,必经的点数就是这两点间简单路径上的圆点的个数。那么这样问题就转化为求树上两点间经过的圆点的个数了。那么我们可以再加一个LCA来解决问题。对于两个圆点,它们的LCA一定是一个方点(当然一个是另一个的祖先时除外)。经过推导可以得出
- ans=(dep[u]+dep[v]-2×dep[LCA(u,v)])/2-1;
然后需要注意的地方就是整个图不一定是全连通的,以及题目中所给的起点终点是两条边,不是点,所以我们把四个顶点都算一遍取最大值。
题本身难度不大,难度在于代码打不出来。。。
代码调了一晚上没调出来结果是数组开小了。。。不说了,还是感谢lc大锅。
#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<vector>
#include<cmath>
using namespace std;
const int maxn=100000+1,maxm=300000+1;
struct Edge{
int next,to;
}edge[maxn<<1];
int head[maxn],len=1;
void Add(int u,int v){
edge[++len].next=head[u];
edge[len].to=v;
head[u]=len;
}
vector<int> G[maxn];
int dfs_clock=0,sta[maxn],top=0,dfn[maxn],low[maxn];
int belong[maxn],dcc_cnt;
bool cut[maxn];
void Tarjan(int u,int fa){
dfn[u]=low[u]=++dfs_clock;
sta[++top]=u;
int son=0;
for(int i=head[u];i;i=edge[i].next){
int v=edge[i].to;
if(!dfn[v]){
son++;
Tarjan(v,fa);
low[u]=min(low[u],low[v]);
if(dfn[u]<=low[v]){
if(u!=fa) cut[u]=1;//习惯性判了个割点,这里不判也没事
dcc_cnt++;
int x;
while(true){
x=sta[top--];
G[dcc_cnt].push_back(x);
G[x].push_back(dcc_cnt);
if(v==x) break;
}
G[dcc_cnt].push_back(u);//一个割点可能同属于多个点双,故不退栈,单独处理
G[u].push_back(dcc_cnt);
}
}
else
low[u]=min(low[u],dfn[v]);
}
if(u==fa&&son>=2) cut[u]=1;//习惯性判了个割点,这里不判也没事
}
int f[maxn][30],dep[maxn];
bool vis[maxn];
void Dfs(int u,int fa){//有毒的倍增法求LCA
dep[u]=dep[fa]+1;
vis[u]=1;
f[u][0]=fa;
for(int i=1;(1<<i)<=dep[u];i++){
f[u][i]=f[f[u][i-1]][i-1];
}
for(int i=0;i<G[u].size();i++){
int v=G[u][i];
if(v!=fa&&!vis[v]) Dfs(v,u);
}
}
int Lca(int s,int t){
if(dep[s]>dep[t])swap(s,t);
int d=dep[t]-dep[s];
int k=0;
while(d){
if(d&1) t=f[t][k];
k++;
d>>=1;
}
if(s==t) return s;
for(int i=20;i>=0;i--){
if(f[s][i]!=f[t][i])
s=f[s][i],t=f[t][i];
}
return f[s][0];
}
int from[maxn],to[maxn];
void Del(int n){//多组数据,初始化
memset(from,0,sizeof(from));
memset(to,0,sizeof(to));
memset(head,0,sizeof(head));
len=0;
memset(G,0,sizeof(G));
memset(vis,0,sizeof(vis));
memset(cut,0,sizeof(cut));
memset(dfn,0,sizeof(dfn));
memset(low,0,sizeof(low));
memset(f,0,sizeof(f));
memset(dep,0,sizeof(dep));
dcc_cnt=n;//方点从n+1开始编号
top=0;
dfs_clock=0;
}
int Calc(int u,int v){
return (dep[u]+dep[v]-(dep[Lca(u,v)]<<1))/2-1;
}
int solve(int xx,int yy){
return (dep[xx]+dep[yy]-2*dep[Lca(xx,yy)])/2-1;
}
int main(){
int n,m;
while(scanf("%d%d",&n,&m) && n!=0){
if(n==0||m==0) return 0;
Del(n);
int u,v;
for(int i=1;i<=m;i++){
scanf("%d%d",&u,&v);
Add(u,v);
Add(v,u);
from[i]=u,to[i]=v;
}
for(int i=1;i<=n;i++){
if(!dfn[i])
Tarjan(i,i);
}
for(int i=1;i<=dcc_cnt;i++){
if(!vis[i])
Dfs(i,0);
}
/*for(int i=1;i<=dcc_cnt;i++){
int u=G[i].size();
for(int j=0;j<u;j++){
printf("%d %d\n",i,G[i][j]);
}
}*/
int q;
scanf("%d",&q);
//printf("%d %d %d\n",Lca(2,3),dep[2],dep[3]);
for(int i=1;i<=q;i++){
scanf("%d%d",&u,&v);
int a1=from[u],a2=to[u],b1=from[v],b2=to[v];
//printf("%d %d %d %d %d %d %d %d %d\n",a1,dep[a1],a2,dep[a2],b1,dep[b1],b2,dep[b2],Lca(b1,b2));
int a=solve(a1,b1),b=solve(a1,b2),c=solve(a2,b1),d=solve(a2,b2);//四个点都试一下
//printf("%d %d %d %d\n",a,b,c,d);
//printf("%d %d %d\n",a2,b1,Lca(a2,b1));
printf("%d\n",max(max(a,b),max(c,d)));
}
}
}
UVA1464 Traffic Real Time Query System的更多相关文章
- CH#24C 逃不掉的路 和 HDU3686 Traffic Real Time Query System
逃不掉的路 CH Round #24 - 三体杯 Round #1 题目描述 现代社会,路是必不可少的.任意两个城镇都有路相连,而且往往不止一条.但有些路连年被各种XXOO,走着很不爽.按理说条条大路 ...
- HDU 3686 Traffic Real Time Query System (图论)
HDU 3686 Traffic Real Time Query System 题目大意 给一个N个点M条边的无向图,然后有Q个询问X,Y,问第X边到第Y边必需要经过的点有多少个. solution ...
- HDU 3686 Traffic Real Time Query System(双连通分量缩点+LCA)(2010 Asia Hangzhou Regional Contest)
Problem Description City C is really a nightmare of all drivers for its traffic jams. To solve the t ...
- HDU3686 Traffic Real Time Query System 题解
题目 City C is really a nightmare of all drivers for its traffic jams. To solve the traffic problem, t ...
- Traffic Real Time Query System 圆方树+LCA
题目描述 City C is really a nightmare of all drivers for its traffic jams. To solve the traffic problem, ...
- HDU Traffic Real Time Query System
题目大意是:对于(n, m)的图,给定边a, b查询从a到b要经过的割点的最少数目. 先tarjan算法求双连通然后缩点,即对于每个割点将周围的每个双连通看成一个点与之相连.然后求解LCA即可,距离d ...
- UVALive-4839 HDU-3686 Traffic Real Time Query System 题解
题目大意: 有一张无向连通图,问从一条边走到另一条边必定要经过的点有几个. 思路: 先用tarjan将双连通分量都并起来,剩下的再将割点独立出来,建成一棵树,之后记录每个点到根有几个割点,再用RMQ求 ...
- 【HDOJ】3686 Traffic Real Time Query System
这题做了几个小时,基本思路肯定是求两点路径中的割点数目,思路是tarjan缩点,然后以割点和连通块作为新节点见图.转化为lca求解.结合点——双连通分量与LCA. /* 3686 */ #includ ...
- hdu 3686 Traffic Real Time Query System 点双两通分量 + LCA。这题有重边!!!
http://acm.hdu.edu.cn/showproblem.php?pid=3686 我要把这题记录下来. 一直wa. 自己生成数据都是AC的.现在还是wa.留坑. 我感觉我现在倒下去床上就能 ...
随机推荐
- [C#.NET 拾遗补漏]02:数组的几个小知识
阅读本文大概需要 1.5 分钟. 数组本身相对来说比较简单,能想到的可写的东西不多.但还是有一些知识点值得总结和知晓一 下.有的知识点,知不知道不重要,工作中用的时候搜索一下就可以了,毕竟实现一个功 ...
- Swagger使用的时候报错:Failed to load API definition
NuGet添加Swashbuckle.AspNetCore,在Startup.cs添加和启用中间件Swagger public void ConfigureServices(IServiceColle ...
- jsp页面用DBHelper实现简单的登陆验证
首先我们需要写一个简单的登陆页面login.jsp,然后用from表单提交给index.jsp页面.在index.jsp页面通过DBHelper连接数据库判断账号和密码,如果密码正确则显示登陆成功. ...
- Istio的运维-诊断工具(istio 系列五)
Istio的运维-诊断工具 在参考官方文档的时候发现环境偶尔会出现问题,因此插入一章与调试有关的内容,便于简单问题的定位.涵盖官方文档的诊断工具章节 目录 Istio的运维-诊断工具 使用istioc ...
- router路由配置
vue项目中router路由配置 介绍 路由:控制组件之间的跳转,不会实现请求.不用页面刷新,直接跳转-切换组件>>> 安装 本地环境安装路由插件vue-router: c ...
- Redis PHP扩展安装步骤
### 下载最新的扩展文件,解压并进入文件夹 wget https://codeload.github.com/phpredis/phpredis/tar.gz/2.2.7 tar -zxvf 2.2 ...
- Flutter 中由 BuildContext 引发的血案
今天和各位分享一个博主在实际开发中遇到的问题,以及解决方法.废话不多说,我们先来看需求: 我们要做一个iOS风格的底部菜单弹出组件,具体涉及showCupertinoModalPopup()方法,该方 ...
- Largest Rectangle in a Histogram【单调栈模板】
Largest Rectangle in a Histogram 题目链接(点击)来源poj 2559 A histogram is a polygon composed of a sequence ...
- 3dTiles 数据规范详解[1] 介绍
版权:转载请带原地址.https://www.cnblogs.com/onsummer/p/12799366.html @秋意正寒 Web中的三维 html5和webgl技术使得浏览器三维变成了可能. ...
- .Net Core微服务入门全纪录(三)——Consul-服务注册与发现(下)
前言 上一篇[.Net Core微服务入门全纪录(二)--Consul-服务注册与发现(上)]已经成功将我们的服务注册到Consul中,接下来就该客户端通过Consul去做服务发现了. 服务发现 同样 ...