poj 3137(强连通)

题目意思 :

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;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值