题目传送门

人生第一道黑题呢,虽然这题是黑题中的水题并且我调了一整节课,但是我还是很兴奋啊.毕竟人生第一道黑题啊

这个题根据题意,先把整个图进行tarjan缩点,建出一棵树,对于每一组询问,两点之间的距离+1就是答案,求出他们的LCA,两点到LCA的距离之和就是两点之间的距离

然后....LCA我采用的是树剖方法,然后对于这个无向图中的强连通分量(其实是叫双连通分量),和有向图中强连通分量的求法并没有什么太大的差别

dfs时,注意不要让它走向父节点就好了

然后,,,,这题剩下的就是调试了

#include <iostream>
#include <cstdlib>
#include <cstdio>
#define min(a,b) (a<b?a:b) using namespace std; const int M=5e4+5;
const int N=1e4+5; struct edge{
int to,next,pre;
}e[(M<<1)],t[(M<<1)]; int n,m,q,dfn[N],low[N],_top;
int siz[N],f[N],son[N],deep[N];
int head[N],tot,top[N],s[N],cnt;
int Tot,Head[N],total,idx[N],anc;
bool ins[N]; inline int read(){
int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){
if(ch=='-') f=-1;
ch=getchar();
}
while(ch>='0'&&ch<='9'){
x=x*10+ch-'0';
ch=getchar();
}
return x*f;
} inline void build_edge(int u,int v){
e[++tot].next=head[u];head[u]=tot;
e[tot].to=v;e[tot].pre=v;return ;
} inline void build_tree(int u,int v){
t[++Tot].next=Head[u];Head[u]=Tot;
t[Tot].to=v;t[Tot].pre=v;return ;
} inline void tarjan(int cur,int anc){
s[++_top]=cur;dfn[cur]=low[cur]=++cnt;ins[cur]=true;
for(register int i=head[cur];i;i=e[i].next){
register int k=e[i].to;
if(k==anc) continue;
if(!dfn[k]){
tarjan(k,cur);
low[cur]=min(low[cur],low[k]);
}else if(ins[k]) low[cur]=min(low[cur],dfn[k]);
}
if(low[cur]==dfn[cur]){
++total;
while(s[_top+1]!=cur){
idx[s[_top]]=total;
ins[s[_top--]]=false;
}
}
return ;
} inline void dfs1(int now,int anc,int dep){
deep[now]=dep;f[now]=anc;siz[now]=1;int maxson=-1;
for(register int i=Head[now];i;i=t[i].next){
register int k=t[i].to;if(k==anc) continue;
dfs1(k,now,dep+1);siz[now]+=siz[k];
if(siz[k]>maxson) maxson=siz[k],son[now]=k;
}
return ;
} inline void dfs2(int now,int topf){
top[now]=topf;if(!son[now]) return ;dfs2(son[now],topf);
for(int i=Head[now];i;i=t[i].next){
int k=t[i].to;
if(k==f[now]||k==son[now]) continue;
dfs2(k,k);
}
return ;
} inline int LCA(int x,int y){
while(top[x]!=top[y]){
if(deep[top[x]]<deep[top[y]]) y=f[top[y]];
else x=f[top[x]];
}
return (deep[x]<deep[y]?x:y);
} inline void print(int x){
if(!x) return ;
print(x>>1);
printf("%d",x&1);
} int main(){
n=read();m=read();
for(register int i=1;i<=m;++i){
register int u,v;
u=read();v=read();
build_edge(u,v);build_edge(v,u);
}
for(register int i=1;i<=n;++i) if(!dfn[i]) tarjan(i,0);
for(register int i=1;i<=n;++i)
for(register int j=head[i];j;j=e[j].next){
register int k=e[j].to;
if(idx[i]!=idx[k])
build_tree(idx[i],idx[k]);//build_tree(idx[k],idx[i]);
}
dfs1(1,0,1);dfs2(1,1);
q=read();
while(q--){
register int u=read(),v=read();
u=idx[u];v=idx[v];anc=LCA(u,v);
register int ans;
ans=deep[u]+deep[v]-(deep[anc]<<1)+1;
print(ans);printf("\n");
}
return 0;
}

LuoGu P2783 有机化学之神偶尔会做作弊的更多相关文章

  1. 【题解】Luogu P2783 有机化学之神偶尔会做作弊

    原题链接:P2783 有机化学之神偶尔会做作弊 一看,是黑题,太毒瘤了,不写 什么单链??! 只会画有机化学中正六边形的我觉得这样不行QAQ(我才初二) 当然,题目也给你了详细的解释 实际呢,这道题先 ...

  2. luogu P2783 有机化学之神偶尔会做作弊 |Tarjan+LCA

    题目背景 XS中学化学竞赛组教练是一个酷爱炉石的人. 有一天他一边搓炉石一边监考,而你作为一个信息竞赛的大神也来凑热闹. 然而你的化竞基友却向你求助了. "第1354题怎么做"&l ...

  3. 洛谷 P2783 有机化学之神偶尔会做作弊 解题报告

    P2783 有机化学之神偶尔会做作弊 题目背景 XS中学化学竞赛组教练是一个酷爱炉石的人. 有一天他一边搓炉石一边监考,而你作为一个信息竞赛的大神也来凑热闹. 然而你的化竞基友却向你求助了. &quo ...

  4. 洛谷 P2783 有机化学之神偶尔会做作弊(Tarjan,LCA)

    题目背景 LS中学化学竞赛组教练是一个酷爱炉石的人. 有一天他一边搓炉石一边监考,而你作为一个信息竞赛的大神也来凑热闹. 然而你的化竞基友却向你求助了. “第1354题怎么做”<--手语 他问道 ...

  5. [洛谷P2783]有机化学之神偶尔会做作弊

    第一次做出来黑题祭 虽然感觉难度其实并不到黑题的难度 题解: 其实这道题并没用什么特别的知识,只是Tarjan求双联通分量和LCA的结合. 所以,我们可以很显然的发现(如此恶劣的词汇,逃 这道题其实就 ...

  6. Tarjan+LCA【洛谷P2783】 有机化学之神偶尔会做作弊

    [洛谷P2783] 有机化学之神偶尔会做作弊 题目背景 XS中学化学竞赛组教练是一个酷爱炉石的人. 有一天他一边搓炉石一边监考,而你作为一个信息竞赛的大神也来凑热闹. 然而你的化竞基友却向你求助了. ...

  7. 【洛谷 P2783】 有机化学之神偶尔会做作弊 (双联通分量)

    题目链接 可能是除了<概率论>的最水的黑题了吧 用\(Tarjan\)缩点(点双联通分量),然后就是树上两点之间的距离了,跑\(LCA\)就好了. #include <cstdio& ...

  8. Tarjan缩点+LCA【p2783】有机化学之神偶尔会做作弊

    Description 你翻到那一题:给定一个烃,只含有单键(给初中生的一个理解性解释:就是一堆碳用横线连起来,横线都是单条的). 然后炎魔之王拉格纳罗斯用他的火焰净化了一切环(???).所有的环状碳 ...

  9. [luogu2783] 有机化学之神偶尔会做作弊

    题目链接 洛谷. Solution 边双缩点然后\(lca\)跑\(dis\)就好了. 注意这里是边双,不知道为啥所有题解都说的是点双. 边双是定义在点上的,即每个点只属于一个边双:点双是定义在边上的 ...

随机推荐

  1. 三十二、Linux 进程与信号——不可靠信号

    32.1 不可靠信号问题 发生信号时关联动作被重置为默认设置 信号可能丢失 程序片段 在进入 sig_int 与再次调用 signal 之间发生的 SIGINT 信号将不会捕获 导致进程终止 以前版本 ...

  2. axis2客户端

    package com.yq.test; /** * Created by yangqing on 2016/12/6. */ import java.util.Date; import java.u ...

  3. 20155324 2016-2017-2 《Java程序设计》第6周学习总结

    20155324 2016-2017-2 <Java程序设计>第6周学习总结 教材学习内容总结 InputStream与OutputStream 串流设计 1.串流:Java将输入/输出抽 ...

  4. mysql日期时间函数(常用的)

    mysql> SELECT NOW();  #返回(打印)当前日期和时间+---------------------+| NOW() |+---------------------+| 2017 ...

  5. [C++]Linux之计算内存利用率与辨析

    声明:如需引用或者摘抄本博文源码或者其文章的,请在显著处注明,来源于本博文/作者,以示尊重劳动成果,助力开源精神.也欢迎大家一起探讨,交流,以共同进步,乃至成为朋友- 0.0 /* @url:http ...

  6. flash 概要分析器

    这个东东 调试用.会每毫秒一次记录正在运行的函数 及相关数据    只在调试版flash player才能用. startSampling  开始记录 getSamples  获取记录 Sample ...

  7. Spring Boot学习记录(二)--thymeleaf模板 - CSDN博客

    ==他的博客应该不错,没有细看 Spring Boot学习记录(二)--thymeleaf模板 - CSDN博客 http://blog.csdn.net/u012706811/article/det ...

  8. 《一头扎进SpringMvc视频教程》

    第二章 SpringMvc控制器 第三章 Rest风格的资源URL 第四章 SpringMvc上传文件

  9. Aizu 2170 Marked Ancestor

    题意:出一颗树,有两种操作:1. mark  u  标记结点u2.query  u  询问离u最近的且被标记的祖先结点是哪个让你输出所有询问的和. 思路:数据量太小,直接暴力dfs就可以了 #incl ...

  10. OsWatcher 使用详解

    软件下载地址: https://support.oracle.com/epmos/faces/DocumentDisplay?_afrLoop=520996062954556&id=30113 ...