第一问应该转化为求入度为0的点个数

第二问取入度为0的点个数与出度为0的点个数的较大值,注意特判仅有一个强连通分量的情况。

参考:题解 P2746 【[USACO5.3]校园网Network of Schools】


首先用Kosaraju求出各个强连通分量缩点,然后求出各个强连通分量间的有向边。 ```cpp const int N = 186; int n, cnt,mark[N],c,r; vi v[N], rv[N], dfn, scc[N]; set s[N], rs[N]; bitset<200> vis;

void dfs(int x)
{
vis[x] = 1;
for (auto it : v[x])
{
if (!vis[it])
dfs(it);
}
dfn.pb(x);
}

void dfs1(int x)
{
scc[cnt].pb(x);
mark[x] = cnt;
for (auto it : rv[x])
{
if (!mark[it])
dfs1(it);
}
}

int main()
{
ios::sync_with_stdio(false);
cin.tie(0);
cin >> n;
f(i, 1, n)
{
int x;
cin >> x;
while (x)
{
v[i].pb(x);
rv[x].pb(i);
cin >> x;
}
}
vis.reset();
f(i, 1, n)
{
if (!vis[i])
{
dfs(i);
}
}
for (auto it = dfn.rbegin(); it != dfn.rend(); it++)
{
if (!mark[*it])
{
cnt += 1;
dfs1(*it);
}
}
f(i, 1, n)
{
for (auto it : v[i])
{
if(mark[i]!=mark[it]){
s[mark[i]].insert(mark[it]);
}
}
for (auto it : rv[i])
{
if(mark[i]!=mark[it]){
rs[mark[i]].insert(mark[it]);
}
}
}
f(i,1,cnt){
if(!s[i].size())c+=1;
if(!rs[i].size())r+=1;
}
cout<<r<<”\n”<<(cnt==1?0:max(c,r));
return 0;
}


此文章已被阅读次数:正在加载...更新于