news 2026/4/28 8:08:55

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

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
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点属于的强连通分量编号
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 16:21:00

LeRobot v0.4.0 正式发布:全面提升开源机器人的学习能力

简要总结LeRobot v0.4.0 为开源机器人领域带来重要升级&#xff1a;引入可扩展的 Datasets v3.0、强大的新 VLA (视觉-语言-动作) 模型如 PI0.5 与 GR00T N1.5&#xff0c;以及全新的插件系统&#xff0c;简化硬件集成。该版本还新增对 LIBERO 与 Meta-World 仿真的支持、简化多…

作者头像 李华
网站建设 2026/4/23 16:52:43

【URP】Unity[后处理]颜色曲线ColorCurves

ColorCurves 是 Unity 通用渲染管线(URP)中的一种高级颜色分级工具&#xff0c;它允许通过曲线精细调整图像的色相、饱和度和亮度。这种工具最初在专业影视后期软件(如 Fusion)中成熟应用&#xff0c;后被引入游戏引擎用于实时渲染的色彩控制。ColorCurves 提供了8条独立曲线&a…

作者头像 李华
网站建设 2026/4/25 18:06:58

微软 Foundry Local - 本地 AI 推理解决方案

软在其 2025 Build 大会上发布了 Foundry Local&#xff0c;能够在本地设备上执行 AI 推理&#xff0c;意味着可以利用本地的 AI 算力&#xff0c;如&#xff1a;CPU/GPU/NPU&#xff1b;也让用户在隐私方面得到了充足的保障&#xff0c;还能有改善成本效益&#xff01;Foundry…

作者头像 李华
网站建设 2026/4/23 21:48:41

Git能上传多大的文件

Git 本身对文件大小没有强制上限&#xff0c;但核心限制来自两个层面&#xff1a;Git 的设计初衷和远程仓库的规则&#xff08;比如 GitHub、GitLab、Gitee 等平台的限制&#xff09;&#xff0c;结合你当前“上传包含嵌套文件夹/子模块”的场景&#xff0c;具体说明如下&#…

作者头像 李华