题目意思 :
N(2<N<100)各学校之间有单向的网络,每个学校得到一套软件后,可以通过单向网络向周边的学校传输,问题1:初始至少需要向多少个学校发放软件,使得网络内所有的学校最终都能得到软件。2,至少需要添加几条传输线路(边),使任意向一个学校发放软件后,经过若干次传送,网络内所有的学校最终都能得到软件。
也就是: 给定一个有向图,求:
1) 至少要选几个顶点,才能做到从这些顶点出发,可以到达全部顶点
2) 至少要加多少条边,才能使得从任何一个顶点出发,都能到达全部顶点
解题思路
1. 求出所有强连通分量(kosaraju算法)
2. 每个强连通分量看成一点,则形成一个有向无环图。
3. 新图上面有多少个入度为0的顶点,问题1的答案就是多少
在新图上要加几条边,使得新图变成强连通的,问题2的答案就是多少。
加边的方法:
要为每个入度为0的点添加入边,为每个出度为0的点添加出边,最少的加边方法就是连接出度和入度为零的点,再补上剩下的出度为零的点边或者是出度为零的点得点;也就是
如果有n个入度为零的点,m个出度为零的点,那么加的变得条数就是max(n,m);
#include <stdio.h>
#include<string.h>
#define MAXN 200
int g[MAXN][MAXN];/*图*/
int in[MAXN],out[MAXN],order[MAXN]/*第二次深搜的顺序*/;
int belong[MAXN];/*各个点所属于的连通分量*/
int vis[MAXN];/*是否遍历*/
int count/*计数器*/, num/*计数器*/ ,i, j, k, n/*点的个数*/;
int max(int a, int b)
{
return a > b ? a : b;
}
void init()/*读数据*/
{
int i, j;
scanf("%d", &n);
for (i = 1; i <= n; i++)
while (scanf("%d", &j) && j)
g[i][j] = 1;
}
void dfs(int v)/*第一次深搜确定点的顺序*/
{
vis[v] = 1;
int i;
for (i = 1; i <= n; i++)
if ((!vis[i]) && g[v][i])
dfs(i);
order[++num] = v;
}
void dfsr(int v)/*第二次深搜确定各个连通分量*/
{
vis[v] = 1;
belong[v] = count;
int i;
for (i = 1; i <= n; i++)
if ((!vis[i]) && g[i][v])
dfsr(i);
}
void kosaraju()
{
for (i = 1; i <= n; i++)//第一次深搜
if (!vis[i])
dfs(i);
memset(vis, 0, sizeof(vis));
for (i = n; i >= 1; i--)//第二次深搜
if (!vis[order[i]])
{
count ++;//统计连通分量个数
dfsr(order[i]);
}
}
void solve()
{
int i, j, sum1 = 0, sum2 = 0;
for (i = 1; i <= n; i++)//计算各个连通分量的 出度和入读
for (j = 1; j <= n; j++)
if (g[i][j] && belong[i] != belong[j])
{
in[belong[j]]++;
out[belong[i]]++;
}
for (i = 1; i <= count; i++)//统计入度和出度为零的点的个数
{
if (!in[i])
sum2++;
if (!out[i])
sum1++;
}
if (count == 1)//特殊情况 原图强连通
printf("1\n0\n");
else
printf("%d\n%d\n", sum2, max(sum2, sum1));
}
int main()
{
init();
kosaraju();
solve();
return 0;
}