P2002メッセージ拡散問題解決
非常に耳を回し縮小裸、メッセージが十分であるポイントに各強連結成分は、エンドポイントを縮小した後、我々が浸透のすべての点を議論しなければならないことは明らかであることは明らかです。数がゼロのポイントに等しくなるように、中度ゼロ点(他の点は彼にニュースを与えない)のために、私たちは、それにメッセージを与える必要があります。データの大きさに、私たちは再ビルドマップする必要はありません。
それは、小さなディテールことを言及する価値があります。複数のエッジとループバック。私は(この点は自己ループQAQがあることも可能である)T点言い渡さ重いエッジマップの結果で開始し、その後、特別な裁判官があまりにもリングから直接マップを宣告削除し、それがこの質問を何に重い副作用があるかのように思えます??(霧
コード:タイトルテンプレート、テンプレートのタイトルのほとんど裸も簡単なポイントを超えると思われます
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <stack>
#include <vector>
#include <map>
#define N 100005
#define ll long long
using namespace std;
inline int read()
{
int x = 0, f = 1;
char c = getchar();
while (c < '0' || c > '9') { if (c == '-') f = -1; c = getchar(); }
while (c >= '0' && c <= '9') { x = x * 10 + c - '0'; c = getchar(); }
return x * f;
}
vector < int > a[N];
stack < int > s;
map < pair < int, int >, int > mp;
int n, m, times, dfn[N], low[N], instc[N], vis[N], mtot, du[N], ans;
//vector存图可能会慢
void tarjan(int x)
{
dfn[x] = low[x] = ++times;
instc[x] = 1;
s.push(x);
for (int i = 0; i < a[x].size(); i++)
{
int u = a[x][i];
if (!dfn[u])
{
tarjan(u);
low[x] = min(low[x], low[u]);
}
else if (instc[u])
low[x] = min(low[x], dfn[u]);
}
if (dfn[x] == low[x])
{
++mtot;
vis[x] = mtot;
instc[x] = false;
while (s.top() != x)
{
vis[s.top()] = mtot;
instc[s.top()] = 0;
s.pop();
}
s.pop();
}
}
int main()
{
n = read(), m = read();
for (int i = 1; i <= m; i++) { int a1 = read(), a2 = read(); if (a1 != a2) a[a1].push_back(a2); }
for (int i = 1; i <= n; i++)
if (!dfn[i]) tarjan(i);
for (int i = 1; i <= n; i++)
{
for (int j = 0; j < a[i].size(); j++)
{
int u = a[i][j];
if (vis[u] != vis[i])
{
du[vis[u]]++; //更新入度 (不在乎是否重复)
}
}
}
for (int i = 1; i <= mtot; i++)
if (du[i] == 0) ans++; //统计入度累加答案
cout << ans << endl;
return 0;
}
最後に、私は、いくつかの同様の質問をお勧めしますtarjan意志ブラシの操作を行います。
P2341 [HAOI2006]人気の牛| [テンプレート]強連結成分
グループの人気はQAQそれをテストしていないので、そう裸tarjanは、青色のタイトルである理由最後Tucao
19.09.05