P3916 图的遍历——反图

给出 \(N\) 个点,\(M\) 条边的有向图,对于每个点 \(v\),求 \(A(v)\) 表示从点 \(v\) 出发,能到达的编号最大的点。

这题有一个巧妙思路,构造反图,把依次找每个能到达的最大的点,转化为从大到小枚举每个点搜索当前点能到达的所有点,第一次找到的点一定是最大点。

反图思路就是建每一条有向边的反向边。这种思想很重要

#include <bits/stdc++.h>

using namespace std;

const int N = 100005;
int n, m;
vector<int>e[N];
int ans[N];
int f;
void dfs(int u)
{
	if (ans[u]) {
		return;
	}
	ans[u] = f;
	for (int v : e[u]) {
		dfs(v);
	}
}

int main()
{
	scanf("%d%d", &n, &m);
	for (int i = 0, a, b; i < m; i++) {
		scanf("%d%d", &a, &b);
		// opposite edge
		e[b].push_back(a);
	}
	for (int i = n; i >= 1; i--) {
		if (!ans[i]) {
			f = i;
			dfs(i);
		}
	}
	for (int i = 1; i <= n; i++) {
		printf("%d ", ans[i]);
	}
	printf("\n");
	return 0;
}
posted @ 2023-01-31 19:52  Vegdie  阅读(76)  评论(0编辑  收藏  举报