Tarjan算法模板

一、最近公共祖先(LCA)

LCALeast Common Ancestor

P3379 【模板】最近公共祖先(LCA)

#include <bits/stdc++.h>using namespace std;
typedef long long ll;ll quickin(void)
{ll ret = 0;bool flag = false;char ch = getchar();while (ch < '0' || ch > '9'){if (ch == '-')    flag = true;ch = getchar();}while (ch >= '0' && ch <= '9' && ch != EOF){ret = ret * 10 + ch - '0';ch = getchar();}if (flag)    ret = -ret;return ret;
}// 存查询的数据结构 
typedef pair<int, int> P;  // first -- 另一个节点; second -- 查询编号 const int MAX = 5e5 + 3;
int N, M, S;               // N -- 节点数; M -- 查询数; S -- 根节点 
vector<int> G[MAX];        // 存树,按无向图存储 
vector<P> query[MAX];      // 存查询,双向存 
int par[MAX], ans[MAX];    // par[] -- 各节点的父节点; ans[] -- 查询答案,按编号存储 
bool vis[MAX];             // vis[] -- 是否访问该节点 // 并查集初始化 
void init(void)
{for (int i = 1; i <= N; ++i)par[i] = i;
}// 并查集查询 
int find(int x)
{if (par[x] == x)    return x;elsereturn par[x] = find(par[x]);
}// tarjan算法 + 并查集 
void tarjan(int u)
{vis[u] = true;                         // 入u, 标记u// 遍历u的每条边 for (int i = 0; i < G[u].size(); ++i){int v = G[u][i];if (!vis[v])                       // 防止访问父节点 {tarjan(v);par[v] = u;                    // 回u, v指向u }}// 离u, 处理查询 for (int i = 0; i < query[u].size(); ++i){P p = query[u][i];int v = p.first, id = p.second;if (vis[v])    ans[id] = find(v);      // 若v被访问过,则v的根节点即所求 }
}int main()
{#ifdef LOCALfreopen("test.in", "r", stdin);#endifN = quickin(), M = quickin(), S = quickin();for (int i = 0; i < N - 1; ++i){int a, b;a = quickin(), b = quickin();G[a].push_back(b);               // 双向存边 G[b].push_back(a);}for (int i = 0; i < M; ++i){int a, b;a = quickin(), b = quickin();query[a].push_back(P(b, i));     // 双向存查询; i -- 查询编号 query[b].push_back(P(a, i));}init();tarjan(S);for (int i = 0; i < M; ++i)cout << ans[i] << endl;return 0;
}

二、强连通分量(SCC)

SCCStrongly Connected Component

B3609 [图论与代数结构 701] 强连通分量

1、基本概念

搜索树:对图深搜时,每个节点仅访问一次,节点按被访问的顺序和访问时经过的有向边组成搜索树
有向边的分类:

  • 树边:搜索树中的边
  • 返祖边:指向祖先节点的边
  • 横插边:右子树指向左子树的边
  • 前向边:指向子孙节点的边

定理1:返祖边与树边必定构成环,横插边可能与树边构成环,前向边无用。
定理2:强连通分量以树的形式存在于搜索树中,每个强连通分量都有一个根,其余节点都在根的子树中。

#include <bits/stdc++.h>using namespace std;
typedef long long ll;ll quickin(void)
{ll ret = 0;bool flag = false;char ch = getchar();while (ch < '0' || ch > '9'){if (ch == '-')    flag = true;ch = getchar();}while (ch >= '0' && ch <= '9' && ch != EOF){ret = ret * 10 + ch - '0';ch = getchar();}if (flag)    ret = -ret;return ret;
}const int MAX = 1e4 + 3;
int N, M;                       // N -- 节点数; M -- 边数 
vector<int> G[MAX];             // 存有向图 
int dfn[MAX], low[MAX], tot;    
/*
dfn[] -- 时间戳,各节点第一次被访问的顺序(也可以起到vis[]的作用) 
low[] -- 从该节点出发,能访问到的最早的时间戳
tot -- 更新时间戳 
*/
stack<int> S;                   // 将节点按时间戳压栈 
bool instk[MAX];                // 该节点是否在栈中 
vector<vector<int>> ans;        // 存储强连通分量 void tarjan(int u)
{// 入u,盖戳,入栈 dfn[u] = low[u] = ++tot;S.push(u), instk[u] = true;// 遍历u的每条边 for (int i = 0; i < G[u].size(); ++i){int v = G[u][i];if (!dfn[v])                       // 未访问过v(在搜索树中,v是u的孩子) {tarjan(v);low[u] = min(low[u], low[v]);  // 回u,更新low }else if (instk[v])                 // 访问过v,且在栈中(在搜索树中,v是u的祖先) {low[u] = min(low[u], dfn[v]);}}// 离u,处理强连通分量 if (dfn[u] == low[u]){vector<int> scc;int t;do{t = S.top();S.pop(), instk[t] = false;scc.push_back(t);} while (t != u);sort(scc.begin(), scc.end());ans.push_back(scc);}
}bool cmp(const vector<int> &a, const vector<int> &b)
{return a[0] < b[0];
}int main()
{#ifdef LOCALfreopen("test.in", "r", stdin);#endifN = quickin(), M = quickin();for (int i = 0; i < M; ++i){int a, b;a = quickin(), b = quickin();G[a].push_back(b);                  // 存储有向边 }// 图未必是连通的 for (int i = 1; i <= N; ++i){if (!dfn[i])tarjan(i);}sort(ans.begin(), ans.end(), cmp);cout << ans.size() << endl;for (int i = 0; i < ans.size(); ++i){for (int j = 0; j < ans[i].size(); ++j){cout << ans[i][j] << ' ';}cout << endl;}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/7822.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【notepad++】使用

1 notepad 下载路径 https://notepad-plus.en.softonic.com/download 2 设置护眼模式 . 设置——语言格式设置——前景色——黑色 . 背景色——RGB &#xff1a;199 237 204 . 勾选“使用全局背景色”、“使用全局前景色” . 保存并关闭

2009-2022年上市公司华证ESG评级评分数据(含细分项)

2009-2022年上市公司华证ESG评级评分数据&#xff08;含细分项&#xff09; 1、时间&#xff1a;2009-2022年 2、来源&#xff1a;华证ESG 3、指标&#xff1a;证券代码、证券简称、综合评级、年度、综合得分、E评级、E得分、S评级、S得分、G评级、G得分 4、范围&#xff1…

PXE 批量安装部署

目录 一、PEX批量部署优点 二、PXE&#xff1a;预启动执行环境 三、搭建PXE远程服务器 要想全自动安装 接下来请看步骤&#xff1a; 一、PEX批量部署优点 规模化&#xff1a;同时装配多台服务器自动化&#xff1a;安装系统 配置各种服务远程实现&#xff1a;不需要光盘&…

Buuctf-Misc题目练习

打开后是一个gif动图&#xff0c;可以使用stegsolve工具进行逐帧看。 File Format:文件格式 Data Extract:数据提取 Steregram Solve:立体试图 可以左右控制偏移 Frame Browser:帧浏览器 Image Combiner:拼图&#xff0c;图片拼接 所以可以知道我们要选这个Frame Browser …

编程新伙伴:如何利用ChatGPT提升代码编写效率

编程是一项既需要逻辑思维又需要创造性的技术活动。而现在&#xff0c;ChatGPT&#xff0c;一款由OpenAI研发的人工智能&#xff0c;可以帮助你更高效地编写代码。这款AI工具被设计成理解和产生人类的自然语言&#xff0c;现在也得以应用于编程领域。那么&#xff0c;ChatGPT如…

SQL查询语句(二)逻辑运算关键字

上一篇文章中我们提到了条件查询除了一些简单的数学符号之外&#xff0c;还有一些用于条件判断的关键字&#xff0c;如逻辑判断 关键字AND,OR,NOT和范围查找关键字BETWEEN,IN等&#xff1b;下面我们来介绍一些这些关键字的用法以及他们所表达的含义。 目录 逻辑运算关键字 AND…

在K8S中,集群可以做哪些优化?

在Kubernetes&#xff08;简称K8s&#xff09;集群中&#xff0c;可以进行多种优化以提升性能、稳定性和资源利用率。以下是一些常见的优化措施&#xff1a; 控制面组件优化&#xff1a; kube-apiserver 高可用与扩展&#xff1a;通过配置多个API服务器实例并使用负载均衡器分发…

用户管理中心——数据库设计用户注册逻辑设计

用户管理中心——数据库设计&用户注册逻辑设计 规整项目目录1. 数据库自动生成器的使用实现基本的数据库操作&#xff08;操作user表&#xff09; 2. 注册逻辑的设计(1) 写注册逻辑(2) 实现(3) 测试代码 3. 遇到的问题 规整项目目录 utils–存放工具类&#xff0c;比如加密…

信息系统架构基本概念及发展_1.概述和发展

信息系统架构&#xff08;Information Systems Architecture&#xff0c;ISA&#xff09;是一种体系结构&#xff0c;它反映了一个政府、企业或事业单位信息系统的各个组成部分之间的关系&#xff0c;以及信息系统与相关业务&#xff0c;信息系统与相关技术之间的关系。 1.信息…

leetCode33. 搜索旋转排序数组

leetCode33. 搜索旋转排序数组 题目思路 此题的特点是&#xff1a;排好序循环的数组&#xff1a;特点&#xff1a;可以分为两个区间&#xff0c;一半升序&#xff0c;一半降序&#xff0c;或者全部升序 我们可以用二分法&#xff0c;找出升序 到降序的那个临界值&#xff0c;并…

基于Springboot 的 Excel表格的导入导出

首先 &#xff0c;引入相关依赖EasyPOI <dependency><groupId>cn.afterturn</groupId><artifactId>easypoi-spring-boot-starter</artifactId><version>4.4.0</version></dependency> 编写实体类&#xff1a; Data AllArgs…

产品需求文档怎么写?超详细的产品需求文档PRD模板来了!

产品需求文档怎么写&#xff1f;如何写一份简洁明了、外行人看了就能秒懂的产品需求文档呢&#xff1f;今天这篇文章&#xff0c;就来和大家分享如何编写一份高质量的产品需求文档 PRD&#xff01; 下图是来自 boardmix 模板社区的「产品需求文档」模板&#xff0c;它给出了一…

2024.05.07作业

#include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);//窗口相关设置this->resize(540,415);this->setFixedSize(540,415);//窗口标题this->setWindowTitle…

C++变量的作用域与存储类型

一 变量的作用域和存储类型 1 变量的作用域(Scope) 指在源程序中定义变量的位置及其能被读写访问的范围分为局部变量(Local Variable)和全局变量(Global Variable) 1&#xff09;局部变量(Local Variable) 在语句块内定义的变量 形参也是局部变量 特点&#xff1a; 生存期是…

用vim或gvim编辑程序

vim其实不难使用&#xff0c;学习一下就好了。简单功能很快学会。它有三种模式&#xff1a;命令模式&#xff0c;编辑模式&#xff0c;视模式。打开时在命令模式。在命令模式下按 i 进入编辑模式&#xff0c;在编辑模式下按<Esc>键退出编辑模式。在命令模式按 :wq 保存文…

Linux —— 信号(3)

Linux —— 信号&#xff08;3&#xff09; Core dump为什么core默认是被关闭的阻塞信号信号其他相关常见概念信号递达信号未决信号阻塞两者的区别信号的结构 信号集操作函数一个简单使用例子sigpending的使用例子 我们今天接着来了解信号&#xff1a; Core dump 大家不知道有…

Linux网络-PXE高效批量网络装机(命令+截图详细版)

目录 一.部署PXE远程安装服务 1.PXE概述 1.1.PXE批量部署的优点 1.2.要搭建PXE网络体系的前提条件 2.搭建PXE远程安装服务器 2.1.修改相关网络配置&#xff08;仅主机模式&#xff09; 2.2.关闭防火墙&#xff08;老规矩&#xff09; 2.3.保证挂载上 2.4.准备好配置文…

(AI Web、ChatGPT Native、Ai Loading、AI Tools、知豆AI)

目录 1、AI Web 2、ChatGPT Native 3、Ai Loading 4、AI Tools 5、知豆AI 1、AI Web

SQL中的top、limit以及rownum

在SQL查询的时候&#xff0c;需要限制输出的条数&#xff0c;然后对于不同的数据库有不同的限制条数的关键字 在SQL Sever中&#xff0c;限制条数的是top&#xff0c;用法如下&#xff1a; SELECT top 10 * FROM table1在MySQL中&#xff0c;限制条数的是limit&#xff0c;用…

呆滞物料规范管理了,问题就好办了

对于制造企业来说&#xff0c;库存是生存和发展的重要保障&#xff0c;过高的库存会占用企业大量的资金和管理成本&#xff0c;影响企业的正常生产&#xff0c;然而多数中小制造企业还在用人工干预管理&#xff0c;如何控制呆滞物料成为仓储管理的一大难题。 什么是呆滞料 呆滞…