当前位置: 首页 > news >正文

Tarjan全家桶系列--强联通分量

强联通分量(SCC)

有向图中的一个​​极大子图​,其中任意两个节点uv都​​互相可达​(即存在u→vv→u的路径),则这个子图为一个强联通分量
Tarjan 算法基于深度优先搜索(DFS),利用 DFS 序dfnlow链 来判断一个节点是否是某个强连通分量的“根”(即最早被访问的节点)。

基本定义:

强连通分量(SCC):有向图中,任意两个节点 u, v 互相可达的最大子图。
dfn[u]:节点 u 被 DFS 第一次访问的时间戳(DFS 序)。
low[u]:从 u 出发,通过若干条边(可以走树边、后向边、横叉边),能到达的所有节点中 最小的 dfn 值。
换句话说:low[u] = min{ dfn[v] | v 从 u 出发可达 }

关键性质:

如果在 DFS 回溯时发现low[u] == dfn[u],说明 u 是当前 SCC 的“根”。
因为从 u 出发无法回到比 u 更早访问的节点。
此时,从栈顶到 u 的所有节点构成一个 SCC。

栈的作用:

DFS 过程中,将访问的节点压入栈。
当找到一个 SCC 的根时,从栈中弹出直到 u,这些节点就属于同一个 SCC。

模板

说明:void Run(int _n,const vector<int>adj[])为传入点的数量,vector邻接表数组,运行求出所有强联通分量 。·vector<int> Get()为获取scc数组,即scc[i]i点属于的强连通分量编号

template<intN>structSCC{vector<int>scc;vector<int>stk;intdfn[N],low[N];constvector<int>*adj;intn,clk,tot;//tot:scc数量voiddfs(intu){++clk;dfn[u]=low[u]=clk;stk.push_back(u);for(intto:adj[u]){if(dfn[to]==-1){dfs(to);low[u]=min(low[u],low[to]);}elseif(scc[to]==-1)low[u]=min(low[u],dfn[to]);}if(low[u]==dfn[u]){intv=-1;++tot;do{v=stk.back();stk.pop_back();scc[v]=tot;}while(v!=u);}}voidRun(int_n,constvector<int>adj[]){n=_n;clk=tot=0;stk.clear();fill(low,low+3+n,-1);fill(dfn,dfn+3+n,-1);scc.assign(n+3,-1);this->adj=adj;for(inti=1;i<=n;++i)if(scc[i]==-1)dfs(i);}vector<int>Get(){returnscc;}};

使用示例

template<intN>structSCC{vector<int>scc;vector<int>stk;intdfn[N],low[N];constvector<int>*adj;intn,clk,tot;//tot:scc数量voiddfs(intu){++clk;dfn[u]=low[u]=clk;stk.push_back(u);for(intto:adj[u]){if(dfn[to]==-1){dfs(to);low[u]=min(low[u],low[to]);}elseif(scc[to]==-1)low[u]=min(low[u],dfn[to]);}if(low[u]==dfn[u]){intv=-1;++tot;do{v=stk.back();stk.pop_back();scc[v]=tot;}while(v!=u);}}voidRun(int_n,constvector<int>adj[]){n=_n;clk=tot=0;stk.clear();fill(low,low+1+n,-1);fill(dfn,dfn+1+n,-1);scc.assign(n+3,-1);this->adj=adj;for(inti=1;i<=n;++i)if(scc[i]==-1)dfs(i);}vector<int>Get(){returnscc;}};constintmaxn=2*1e5+20;SCC<maxn>T;vector<int>e[maxn];intmain(){intn,m;//n个点,m条边for(inti=1;i<=m;++i){//输入m条边intu,v;cin>>u>>v;e[u].push_back(v);}T.Run(n,e);//跑tarjanvector<int>scc=T.Get();//获取scc数组,scc[i]=i点属于的强连通分量编号
http://www.jsqmd.com/news/83554/

相关文章:

  • 学Simulink——基于高比例可再生能源渗透的复杂电网建模场景实例:大规模光伏并网对区域电网频率稳定影响研究
  • 485报文订阅服务
  • 毕设开源 深度学习火焰检测识别(源码+论文)
  • 【Spring框架】SpringJDBC
  • 校徽批评,何时从“找茬”走向“建设”?——兼评一篇公众号文章的逻辑
  • 中小诊所系统通常具备哪些功能?
  • 【URP】Unity[后处理]颜色曲线ColorCurves
  • 基于Uniapp的手机维修交流小程序
  • 大模型通义千问3-VL-Plus - 视觉推理(本地图片)
  • Profinet转Modbus TCP工业数据采集网关:实现1200PLC 与打标卡数据实时传输
  • Git能上传多大的文件
  • Burp Suite安装保姆级教程,Burp Suite的基本介绍及使用,收藏这一篇就够了
  • [STM8]学习日志-STM8介绍
  • 【渗透测试零基础入门】搭建 DVWA 靶场保姆级教程(超详细),收藏这一篇就够了!_dvwa靶场搭建
  • 一文详解JUC中乐观锁的实现原理(CAS)、实现方式、优缺点及应用场景总结
  • 3.7 Elasticsearch-查询性能剖析:profile API、DFS query_then_fetch
  • 国内纸纱线FSC春夏14至16针,实力公司推荐排行榜揭秘
  • 参数估计(三)-- 隐参数模型
  • 打CTF,逆向分析攻略!
  • LightModel
  • 数据结构入门:从“是什么”到“为什么要学”
  • 当下的网络安全行业前景到底怎么样?还能入行分蛋糕吗?
  • 国内专业纸纱线FSC春夏14-16针工厂,这份推荐榜单别错过
  • 不同扩散模型下煤层瓦斯运移的Comsol数值模拟:双孔扩散及时变扩散模型文献复现
  • 黑神话 地图APP 程序
  • 【笔记篇】【硬件基础篇】电力电子元器件应用手册 阅读笔记(1)电阻器及其应用
  • 双向buck/boost电路仿真(VDCM控制/电压电流双闭环控制) 利用了传统电机的阻尼和旋...
  • OKR与绩效考核结合:优势、挑战与实践路径
  • AD学习笔记-31 DRC检查
  • Linux系统编程——进程进阶:父子关系、终止与资源回收