tarjan在图论中还是挺重要的.这里就简要的梳理一下tarjan的知识点.

tarjan算法与无向图连通性.

首先说一下图中割点和桥的定义.

桥:也称割边,定义类似,在无向图中,若去掉某条边,导致整张图不连通,则该边为割边.

割点:在无向图中,若去掉某个点,导致整张图不连通,则该点为割点.

其他的什么基础知识就不多说了,这里给出桥和割点的判定法则.

割边:dfn[x]<low[y].

感性的理解下,low[y]说明y向下走,没办法通过非树边到达x及以上的点.代码中的小细节就是tarjan时记录过来的边,防止重边对答案的影响.

割边:dfn[N],low[N],bridge[N],num;
inline void tarjan(int x,int in_edge)
{
dfn[x]=low[x]=++num;
for(int i=link[x];i;i=a[i].next)
{
int y=a[i].y;
if(!dfn[y])
{
tarjan(y,i);
low[x]=min(low[x],low[y]);
if(low[y]>dfn[x]) bridge[i]=bridge[i^1]=true;
}
else if(i!=(in_edge^1)) low[x]=min(low[x],dfn[y]);
}
}

割点:

1.若x不是搜索树的根结点,则满足dfn[x]<=low[y].(感性的理解,y只能到达x,无法与x以上的点取得联系)

2.若x是搜索树的1根结点,则满足至少存在两个以上节点才行.(根只有一个儿子显然不行.)

割点:num,dfn[N],low[N],vis[N];
inline void tarjan(int x)
{
dfn[x]=low[x]=++num;
int flag=0;
for(int i=link[x];i;i=a[i].next)
{
int y=a[i].y;
if(!dfn[y])
{
tarjan(y);
low[x]=min(low[x],low[y]);
if(low[y]>=dfn[x])
{
flag++;
if(x!=root||flag>=2) vis[x]=1;
}
}
else low[x]=min(low[x],dfn[y]);
}
}

之后是无向图的双连通分量.

点双联通图:若一个图中不存在割点,则称该图是点双联通图.

点双联通分量:无向图中的极大点双连通图.

边双联通图:若一个图中不存在桥,则称该图是边双联通图.

边双连通分量:无向图中的极大边双联通图.

先讨论边双的情况(因为简单).

判定:一个图是边双连通图的充要条件,图中任意一条边都至少存在一个简单环中.

很显然吧,若不在环中,则该边连接的两个点就断开了,则存在割边,不符合定义.

给出边双联通分量,即缩点的代码:

inline void tarjan(int x,int in_edge)
{
dfn[x]=low[x]=++num;
for(int i=link[x];i;i=a[i].next)
{
int y=a[i].y;
if(!dfn[y])
{
tarjan(y,i);
if(low[y]>dfn[x]) bridge[i]=bridge[i^1]=true;
}
else if(i!=(in_edge^1)) low[x]=min(low[x],dfn[y]);
}
}
inline void dfs(int x)
{
c[x]=dcc;
for(int i=link[x];i;i=a[i].next)
{
int y=a[i].y;
if(c[y]||bridge[i]) continue;
dfs(y);
}
}
main函数内:
for(int i=1;i<=n;++i) if(!c[i]) dcc++,dfs(i);
for(int i=2;i<=tot;++i)
{
int x=a[i^1].y,y=a[i].y;
if(c[x]!=c[y]) add_c(c[x],c[y]);
}

点双连通分量的判定(至少满足一个):

1.图的顶点数不超过2.

2.图中任意两个节点都同时包含在至少一个简单环中.

证明略过.....(还是太菜了..)

点双的代码:

inline void tarjan(int x)
{
dfn[x]=low[x]=++num;
stack[++top]=x;
if(x==root&&link[x]==0)
{
dcc[++cnt].push_back(x);
return;
}
int flag=0;
for(int i=link[x];i;i=a[i].next)
{
int y=a[i].y;
if(!dfn[y])
{
tarjan(y);
low[x]=min(low[x],low[y]);
if(low[y]>=dfn[x])
{
++flag;
if(x!=root||flag>=2) vis[x]=1;
cnt++;
int z=0;
while(z!=y)
{
z=stack[top--];
dcc[cnt].push_back(z);
}
dcc[cnt].push_back(x);
}
}
}
}
main函数内:
num=cnt;
for(int i=1;i<=n;++i) if(vis[i]) new_id[i]=++num;
tc=1;
for(int i=1;i<=cnt;++i)
{
for(int j=0;j<dcc[i].size();++j)
{
int x=dcc[i][j];
if(vis[x])
{
add_c(i,new_id[x]);
add_c(new_id[x],i);
}
else c[x]=i;
}
}

放一道边双缩点的题(码量很大啊...)

364. 网络

最新文章

  1. .net面试题集锦
  2. 【转】mysql忘记密码(未初始化)
  3. 怎样在myEclipse中使用debug调试程序?
  4. C#winform控制textbox输入只能为数字
  5. iOS 自定义UIButton(图片和文字混合)
  6. python-haproxy作业讲解视频总结
  7. PHP 中安装memcache扩展文件下载对应地址。
  8. struts (二)
  9. RTP、RTCP
  10. button 禁止
  11. 非常好的在网页中显示pdf的方法
  12. 本地搜索神器-Everything
  13. android 几个小技巧
  14. Docker: 限制容器可用的 CPU
  15. CSS绝对定位元素居中的几种方法
  16. 「CodeForces - 50C 」Happy Farm 5 (几何)
  17. day45 jQuery
  18. java工程师-面试知识点总结
  19. TCP网络编程
  20. js-设计模式学习笔记-策略模式

热门文章

  1. [NOIP2015 普及组] 扫雷游戏
  2. 小学生都能读懂的网络协议之:WebSocket
  3. php nginx 路径批量配置
  4. MySQL 服务无法启动。 服务没有报告任何错误。 请键入 NET HELPMSG 3534 以获得更多的帮助。
  5. PHP ASCII 排序方法
  6. bzoj#4423-[AMPPZ2013]Bytehattan【并查集】
  7. Springboot2.0整合Redis(注解开发)
  8. ASP.NET Core 学习笔记 第二篇 依赖注入
  9. Notepad++离线安装使用Markdown插件
  10. JavaEE &amp; Tomcat 介绍