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.留坑. 我感觉我现在倒下去床上就能 ...
随机推荐
- TZOJ 车辆拥挤相互往里走
102路公交车是crq经常坐的,闲来无聊,他想知道最高峰时车上有多少人,他发现这辆车只留一个门上下人,于是他想到了一个办法,上车时先数一下车上人员数目(crq所上的站点总是人不太多),之后就坐在车门口 ...
- Babel 7 安装与配置
Babel:帮我们把高级的语法(ES6)转为低级的语法 /* Babel 7.x版本 安装如下 (cnpm i @babel/cli -D) //Bab ...
- 容器技术之Docker私有镜像仓库docker-distribution
在前边的博客中我们说到docker的架构由docker客户端.服务端以及仓库组成:docker仓库就是用来存放镜像的地方:其实docker registry我们理解为存放docker镜像仓库的仓库比较 ...
- CentOS7——配置阿里云镜像源
CentOS7--配置阿里云镜像源 #下载CentOS 7的repo文件 curl -o /etc/yum.repos.d/CentOS-Base.repo http://mirrors.aliyun ...
- 若linux 的分区硬盘满,如何处理?
一.确定是不是真的是磁盘空间不足 输入命令:df –lh 查看磁盘信息 二.如何定位最大文件目录 输入命令:cd / 进入根目录. 输入命令:du -h max-depth=1 寻找当前目录,哪个文件 ...
- (六)maven 聚合和继承
项目目录 my_test 聚合pom <?xml version="1.0" encoding="UTF-8"?> <project xmln ...
- VSCode + WSL 2 + Ruby环境搭建详解
vscode配置ruby开发环境 vscode近年来发展迅速,几乎在3年之间就抢占了原来vim.sublime text的很多份额,犹记得在2015-2016年的时候,ruby推荐的开发环境基本上都是 ...
- Spring Boot 2 实战:利用Redis的Geo功能实现查找附近的位置
1. 前言 老板突然要上线一个需求,获取当前位置方圆一公里的业务代理点.明天上线!当接到这个需求的时候我差点吐血,这时间也太紧张了.赶紧去查相关的技术选型.经过一番折腾,终于在晚上十点完成了这个需求. ...
- Quartz.Net系列(七):Trigger之SimpleScheduleBuilder详解
所有方法图 1.SimpleScheduleBuilder RepeatForever:指定触发器将无限期重复. WithRepeatCount:指定重复次数 var trigger = Trigge ...
- 性能测试之JVM的故障分析工具VisualVM
VisualVM 是随JDK一同发布的jvm诊断工具,通过插件可以扩展很多功能,插件扩展也是其精华所在. 提供了一个可视界面,用于在Java应用程序在Java虚拟机上运行时查看有关Java应用程序的详 ...