题目地址:http://poj.org/problem?id=1236
题目意思:
给定一个有向图,求:
1) 至少要选几个顶点,才能做到从这些顶点出
发,可以到达全部顶点
2) 至少要加多少条边,才能使得从任何一个顶
点出发,都能到达全部顶点
解:
1)缩点
2) DAG上面有多少个入度为0的顶点就是1)的答案
3)假定有 n 个入度为0的点,m个出度为0的点,max(m,n)就是第二个问题的解
Korasaju
#include<iostream>
#include<cstdio>
#include<vector>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn=100+10;
bool vis[maxn];
int ID[maxn]; //点的颜色编号
int ncolor;
vector<int> s;
vector<vector<int> > color(maxn); //同一种颜色的点
vector<vector<int> > G(maxn);
vector<vector<int> > GT(maxn);
void dfs(int u){vis[u]=true;for(int i=0;i<G[u].size();i++){int v=G[u][i];if(!vis[v]) dfs(v);}s.push_back(u);
}
void dfs2(int u){vis[u]=true;ID[u]=ncolor;color[ncolor].push_back(u);for(int i=0;i<GT[u].size();i++){int v=GT[u][i];if(!vis[v]) dfs2(v);}
}
void Korasaju(int n)
{memset(vis,false,sizeof(vis));s.clear();for(int i=1;i<=n;i++) if(!vis[i]) dfs(i);memset(vis,false,sizeof(vis));ncolor=0;for(int i=s.size()-1;i>=0;i--) {int j=s[i];if(!vis[j]) ncolor++,dfs2(j);}
}
void solve()
{int m=0,ans; //m标记入度为0的点有几个 for(int i=1;i<=ncolor;i++){bool ok=true;for(int j=0;j<color[i].size()&&ok;j++){int u=color[i][j];for(int k=0;k<GT[u].size()&&ok;k++){int v=GT[u][k];if(ID[u]!=ID[v]) //有出度 ok=false; }}if(ok) m++;}int n=0; //n标记出度为0的点有几个 for(int i=1;i<=ncolor;i++){bool ok=true;for(int j=0;j<color[i].size()&&ok;j++){int u=color[i][j];for(int k=0;k<G[u].size()&&ok;k++){int v=G[u][k];if(ID[u]!=ID[v]) //有出度 ok=false; }}if(ok) n++;}if(ncolor==1) cout<<1<<endl<<0; //注意,只有一个强连通分支的时候,不需要边 else cout<<m<<endl<<max(m,n);
}int main()
{int n;cin>>n;for(int u=1;u<=n;u++){int v;while(cin>>v&&v){G[u].push_back(v);GT[v].push_back(u);}}Korasaju(n);solve();return 0;
}
Tarjan
#include<iostream>
#include<cstdio>
#include<vector>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn=100+10;
bool vis[maxn];
int ID[maxn]; //????????±à??
int ncolor,index;
vector<int> st,dfn(maxn),low(maxn);
vector<vector<int> > color(maxn); //??????????????
vector<vector<int> > G(maxn);
vector<vector<int> > GT(maxn);
void Tarjan(int u)
{dfn[u]=low[u]=++index;vis[u]=true;st.push_back(u);for(int i=0;i<G[u].size();i++){int v=G[u][i];if(!vis[v]){Tarjan(v);low[u]=min(low[u],low[v]);}else if(find(st.begin(),st.end(),v)!=st.end()) //in stacklow[u]=min(low[u],dfn[v]);}if(dfn[u]==low[u]){int v; ncolor++;do{v=st.back(); st.pop_back();color[ncolor].push_back(v);ID[v]=ncolor;}while(v!=u);}
}
int getGraph(int n)
{index=ncolor=0;for(int i=1;i<=n;i++)if(!vis[i]) Tarjan(i);
}
void solve()
{int m=0,ans; //m±ê????????0?????????? for(int i=1;i<=ncolor;i++){bool ok=true;for(int j=0;j<color[i].size()&&ok;j++){int u=color[i][j];for(int k=0;k<GT[u].size()&&ok;k++){int v=GT[u][k];if(ID[u]!=ID[v]) //?????? ok=false; }}if(ok) m++;}int n=0; //n±ê????????0?????????? for(int i=1;i<=ncolor;i++){bool ok=true;for(int j=0;j<color[i].size()&&ok;j++){int u=color[i][j];for(int k=0;k<G[u].size()&&ok;k++){int v=G[u][k];if(ID[u]!=ID[v]) //?????? ok=false; }}if(ok) n++;}if(ncolor==1) cout<<1<<endl<<0; //×??????????????????¨·??§???±?ò?????è??±? else cout<<m<<endl<<max(m,n);
}int main()
{int n;cin>>n;for(int u=1;u<=n;u++){int v;while(cin>>v&&v){G[u].push_back(v);GT[v].push_back(u);}}getGraph(n);solve();return 0;
}