建筑专业网站有哪些,无锡建设招标网站,青岛模板做网站,网页设计实验报告怎么写概念和性质#xff1a; 强连通#xff1a;在有向图G中#xff0c;如果两个点u和v是互相可达的#xff0c;即从u出发可以到达v#xff0c;从v出发也可以到达u#xff0c;则成u和v是强连通的。
强连通分量#xff1a;如果一个有向图G不是强连通图#xff0c;那么可以把它…概念和性质 强连通在有向图G中如果两个点u和v是互相可达的即从u出发可以到达v从v出发也可以到达u则成u和v是强连通的。
强连通分量如果一个有向图G不是强连通图那么可以把它分成躲个子图其中每个子图的内部是强连通的而且这些子图已经扩展到最大不能与子图外的任一点强连通成这样的一个“极大连通”子图是G的一个强连通分量SCC。
强连通分量的一些性质
1一个点必须有出度和入度才会与其他点强连通。
2把一个SCC从图中挖掉不影响其他点的强连通性。 求强连通分量的方法
Kosaraju算法
算法原理及步骤
1将有向图G的所有边反向建立返图rG反图rG不会改变原图G的强连通性也就不会改变SCC的数量。
2对原图G做一次DFS确定各点的先后顺序可以用vector数组来记录顺序。
3确定了顺序之后在反图上做DFS按顺序从优先级最高的点开始到最低的点。
为什么要在反图上做DFS这样做可以求得被隔离的岛。我们以下图为例图a为原图G图b为反图rG我们将每个SCC都打上阴影将其想象成一个个岛屿也可以理解为一个个缩点那么我们可以发现{a,b,e}这个SCC与其他SCC之间只有出边没有入边所以这个SCC在第二次DFS中是要首先被遍历的 那么我们在反图上遍历它的好处就是可以发现在反图中{a,b,e}这个SCC变成了只有入边没有出边所以我们的遍历一定只会被限制在当前的SCC中确定了第一个SCC。当我们遍历完这个SCC我们将其删除在代码中可以体现为将其标记为已经遍历然后继续从剩下的优先级最高的点开始搜索这一次从c开始搜索因为反边这次搜索也被限制在{c,d}内确定了第二个SCC删除这个SCC然后按照这个步骤确定剩下的SCC。 参考代码
题目来自hdu 1269标准的模板题判断整个图是否强连通求出SCC数量是否为1即可。
在进行第一次dfs确定先后顺序时我们有在递归进入时和递归返回时标记两种方法。分别给出代码
#include bits/stdc.h
using namespace std;
#define endl \n
typedef long long ll;
typedef unsigned long long ull;
const int maxn 1e4 10;
const int INF 0x3fffffff;
const int mod 1000000007;
vectorint G[maxn], rG[maxn]; // 原图G和它的反图
vectorint S; // 存储第一次dfs的结果标记点的先后顺序
int vis[maxn]; // 访问标记
int sccno[maxn]; // sccno[i]表示第i个点所属的强连通分量
int cnt; // 强连通分量的个数
int n, m;void dfs1(int u) { // 在原图G上做一次dfs标记点的先后顺序S.push_back(u); // 记录点的先后顺序vis[u] true;for (int i 0; i G[u].size(); i) {if (!vis[G[u][i]]) {dfs1(G[u][i]);}}
}void dfs2(int u) { // 在反图rG上做一次dfs顺序从标记最大的点开始到标记最小的点。sccno[u] cnt;for (int i 0; i rG[u].size(); i) {if (!sccno[rG[u][i]]) {dfs2(rG[u][i]);} }
}void Kosaraju() {cnt 0;S.clear();memset(sccno, 0, sizeof sccno);memset(vis, 0, sizeof vis);for (int i 1; i n; i) {if (!vis[i]) {dfs1(i);}}for (int i 0; i n; i) {if (!sccno[S[i]]) {cnt;dfs2(S[i]);}}
}void solve() {int u, v;while (cin n m, n ! 0 || m ! 0) {for (int i 1; i n; i) {G[i].clear();rG[i].clear();}for (int i 0; i m; i) {cin u v;G[u].push_back(v); // 原图rG[v].push_back(u); // 反图}Kosaraju();cout (cnt 1 ? Yes\n : No\n);}
}int main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);cout fixed;cout.precision(18);solve();return 0;
} #include bits/stdc.h
using namespace std;
#define endl \n
typedef long long ll;
typedef unsigned long long ull;
const int maxn 1e4 10;
const int INF 0x3fffffff;
const int mod 1000000007;
vectorint G[maxn], rG[maxn]; // 原图G和它的反图
vectorint S; // 存储第一次dfs的结果标记点的先后顺序
int vis[maxn]; // 访问标记
int sccno[maxn]; // sccno[i]表示第i个点所属的强连通分量
int cnt; // 强连通分量的个数
int n, m;void dfs1(int u) { // 在原图G上做一次dfs标记点的先后顺序vis[u] true;for (int i 0; i G[u].size(); i) {if (!vis[G[u][i]]) {dfs1(G[u][i]);}}S.push_back(u); // 记录点的先后顺序标记大的放在S的后面
}void dfs2(int u) { // 在反图rG上做一次dfs顺序从标记最大的点开始到标记最小的点。sccno[u] cnt;for (int i 0; i rG[u].size(); i) {if (!sccno[rG[u][i]]) {dfs2(rG[u][i]);} }
}void Kosaraju() {cnt 0;S.clear();memset(sccno, 0, sizeof sccno);memset(vis, 0, sizeof vis);for (int i 1; i n; i) {if (!vis[i]) {dfs1(i);}}for (int i n - 1; i 0; i--) {if (!sccno[S[i]]) {cnt;dfs2(S[i]);}}
}void solve() {int u, v;while (cin n m, n ! 0 || m ! 0) {for (int i 1; i n; i) {G[i].clear();rG[i].clear();}for (int i 0; i m; i) {cin u v;G[u].push_back(v); // 原图rG[v].push_back(u); // 反图}Kosaraju();cout (cnt 1 ? Yes\n : No\n);}
}int main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);cout fixed;cout.precision(18);solve();return 0;
} Tarjan算法
算法原理及步骤
在讲Tarjan算法之前我们需要给出一个定理一个SCC从其中任意一点出发都至少有一条路径可以回到自己。那么其实从任意一点开始DFS这个点都会成为这个SCC的祖先。
我们可以在DFS求low值的同时把点按SCC有相同的low值分开用栈分离不同的SCC。关于low值的定义可以看看我之前关于割点割边的随笔。
大体步骤
dfn[u]表示dfs时达到顶点u的次序号时间戳low[u]表示以u为根节点的dfs树中次序号最小的顶点的次序号所以当dfn[u]low[u]时以u为根的搜索子树上所有节点是一个强连通分量。 先将顶点u入栈dfn[u]low[u]idx,扫描u能到达的顶点v如果v没有被访问过则dfs(v)low[u]min(low[u],low[v])如果v在栈里low[u]min(low[u],dfn[v])扫描完v以后如果dfn[u]low[u]则将u及其以上顶点出栈。
考虑最先入栈的点每进入一个新的SCC访问并入栈的第1个点都是这个SCC的祖先它的num值和low值相等这个SCC中所有的点的low值都与它相等。 参考代码
例题仍然是hdu1269
#include bits/stdc.h
using namespace std;
#define endl \n
typedef long long ll;
typedef unsigned long long ull;
const int maxn 1e4 10;
const int INF 0x3fffffff;
const int mod 1000000007;
int num[maxn]; // 记录每个节点的dfs次序时间戳
int low[maxn]; // low[v]表示v以及v的后代能退回到的节点的num值最小是多少
int sccno[maxn]; // sccno[u]表示u点属于哪个强连通分量
int st[maxn], top; // 模拟栈
vectorint G[maxn]; // 图
int dfn; // dfs次序时间戳
int cnt; // 强连通分量的个数
int n, m;void dfs(int u) {st[top] u; // u入栈low[u] num[u] dfn;for (int i 0; i G[u].size(); i) {int v G[u][i];if (!num[v]) { // 未访问过的点继续dfsdfs(v); // dfs的最底层是最后一个SCClow[u] min(low[u], low[v]);} else if (!sccno[v]) { // 处理回退边low[u] min(low[u], num[v]);}}if (low[u] num[u]) { // 栈底的点是SCC的祖先它的low numcnt;while (1) {int v st[--top]; // v弹出栈sccno[v] cnt;if (u v) { // 栈底的点是SCC的祖先break;}}}
}void tarjan() {cnt top dfn 0;memset(sccno, 0, sizeof sccno);memset(num, 0, sizeof num);memset(low, 0, sizeof low);for (int i 1; i n; i) {if (!num[i]) {dfs(i);}}
}void solve() {int u, v;while (cin n m, n ! 0 || m ! 0) {for (int i 1; i n; i) {G[i].clear();}for (int i 0; i m; i) {cin u v;G[u].push_back(v);}tarjan();cout (cnt 1 ? Yes\n : No\n);}
}int main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);cout fixed;cout.precision(18);solve();return 0;
}