P2812 校园网络【[USACO]Network of Schools加强版】
题目背景
浙江省的几所OI强校的神犇发明了一种人工智能,可以AC任何题目,所以他们决定建立一个网络来共享这个软件。但是由于他们脑力劳动过多导致全身无力身体被♂掏♂空,他们来找你帮助他们。
题目描述
共有n所学校(n<=10000)已知他们实现设计好的网络共m条线路,为了保证高速,网络是单向的。现在请你告诉他们至少选几所学校作为共享软件的母机母鸡,能使每所学校都可以用上。再告诉他们至少要添加几条线路能使任意一所学校作为母机母鸡都可以使别的学校使用上软件。
输入输出格式
输入格式:
第一行一个整数n。
接下来n行每行有若干个整数,用空格空格隔开。
第i-1行的非零整数x,表示从i到x有一条线路。以0作为结束标志。
输出格式:
第一行一个整数表示问题1的答案。
第二行回答问题2.
我们当然选择在入度为0的点安放母鸡,因为没人能管它们。
但仅仅在入度为0的点放是不够的,因为若构成了一个进不去的环,同样也是不行的。
那行,先tarjan把环缩掉,则入度为0的点之和即为答案1。
答案2我们依旧贪心来想。
我们把出度为0的点连到入度为0的点上去,就构成了强连通分量。
然后把剩下的多的出度或者入度为0的点的个数加上去就行。
code:
#include <cstdio>
int max(int x,int y){return x>y?x:y;}
int min(int x,int y){return x<y?x:y;}
const int N=10010;
int n;
struct node
{
int to,next;
}edge0[N*5];
int head0[N],cnt0=0;
int add0(int u,int v)
{
edge0[++cnt0].next=head0[u];edge0[cnt0].to=v;head0[u]=cnt0;
}
int s[N],tot=0;
void push(int x){s[++tot]=x;}
void pop(){tot--;}
int dfn[N],low[N],index=0,time=0,vis[N],in[N],out[N],ha[N];
void tarjan(int now)
{
push(now);
vis[now]=1;
dfn[now]=low[now]=++time;
for(int i=head0[now];i;i=edge0[i].next)
{
int v=edge0[i].to;
if(!dfn[v]) {tarjan(v);low[now]=min(low[v],low[now]);}
else if(vis[v]) {low[now]=min(dfn[v],low[now]);}
}
if(dfn[now]==low[now])
{
index++;
while(s[tot]!=now)
{
ha[s[tot]]=index;
vis[s[tot]]=0;
pop();
}
ha[s[tot]]=index;
vis[s[tot]]=0;
pop();
}
}
int main()
{
scanf("%d",&n);
int V;
for(int i=1;i<=n;i++)
{
scanf("%d",&V);
while(V)
{
add0(i,V);
scanf("%d",&V);
}
}
for(int i=1;i<=n;i++)
if(!dfn[i])
tarjan(i);
for(int i=1;i<=n;i++)
for(int j=head0[i];j;j=edge0[j].next)
{
int v=edge0[j].to;
if(ha[i]!=ha[v])
{
out[ha[i]]++;
in[ha[v]]++;
}
}
int ans1=0,ans2=0;
for(int i=1;i<=index;i++)
{
if(!in[i]) ans1++;
if(!out[i]) ans2++;
}
printf("%d\n%d\n",ans1,max(ans1,ans2));
return 0;
}
2018.6.6