码迷,mamicode.com
首页 > 其他好文 > 详细

2017.12.23 浙江工大 迎新赛

时间:2017-12-24 12:50:33      阅读:105      评论:0      收藏:0      [点我收藏+]

标签:大神   联通   输入   分析   1.0   输出   efi   表示   故障   

A .

题目描述

        小杰是学院中公认的计算机大神,有一天小杰受邀去机房帮忙接网线,他三下五除二地就帮忙完成了。可是善于思考地他仍意有未尽,他觉得这次的线路有许多奇妙的性质,故来请教你。
        机房有n台不同的主机(编号为0,1,2,?,n?1),共连有n条网线,每台主机恰连着两条网线,通向其他的主机。相互之间有网线相连的主机可以通信,主机可以转发其他主机的信息,换句话说,如果主机A和主机C可以通信,主机B和主机C可以通信,那么主机A和主机B可以通信。现在想在一些主机间添加一些网线,使得任意一个主机故障时,其他的主机之间仍能互相通信(注:在一个主机故障时,所有与该主机直接相连的线路将无法使用)。那么问题来了,在添加的网线最少时,请问有多少种连线的方案,由于连线方案可能非常多,请将方案数对1000,000,007取模。

输入描述:

第一行表示主机数n。
接下来有n行,第i(i=0,1,2,?,n?1)行有两个数a,b,表示主机i和主机a,主机i和主机b各有网线相连。
输入有多组数据,n=0时,表示输入结束,请不要有对该组数据任何输出。

输出描述:

每组数据输出一行,表示方案数对1000,000,007取模的结果。(行末不要有多余的空白字符)
示例1

输入

5
2 3
4 4
0 3
0 2
1 1
12
11 11
10 10
9 9
8 8
7 7
6 6
5 5
4 4
3 3
2 2
1 1
0 0
2
1 1
0 0
0

输出

6
3
0

备注:

2≤n≤3×105,
不超过10组数据 n≥2×105,
不超过20组数据 n≥104,
总数据组数不超过200
 
题目分析 : 其实题目就是让找有多少个联通的图,并且找到每个独立的图中有多少个点,要将所有的图都连通,且保证当只有一个主机坏的时候,其他主机仍保持连通,则可以从每个独立的块中选择两个点出来去进行相连,C(n, 2),每面对两个独立的图,则会在乘以 2 ,因为可以交换点的顺序么,然后再是所有独立连通图的排列顺序,有(n-1)! 种 。
 
补充两个很坑的点 !!!
并查集的路径压缩部分,以前学的版本他路径压缩是不完全的 !! , 必须得最后在遍历一遍补充一些点进行路径压缩 !!!
第二点就是 long long 的地方, 拼经验的地方, long long  ans 要先给大数值 !!! 虽然还不知道为什么 !!
 
#define ll long long
const int eps = 3e5+5;
const ll mod = 1000000007;
const double pi = acos(-1.0);
const int inf = 1<<29;
#define Max(a,b) a>b?a:b
#define Min(a,b) a>b?b:a


int f[eps];
int num[eps];
int n;
ll jc[eps];
ll fang[eps];

int fid(int x){
    //return x==f[x]?x:fid(f[x]);
    int r = x;
    while(r != f[r]){
        r = f[r];
    }
    
    int p = x;
    while(p != r){
        int t = f[p];
        f[p] = r;
        p = t;
    }
    return r;
}

void fun(){
    fang[0] = 1;
    jc[0] = 1;
    for(ll i = 1; i <= 300000; i++){
        jc[i] = jc[i-1]*i;
        fang[i] = fang[i-1]*2;
        jc[i] %= mod;
        fang[i] %= mod;
    }
}

int main() {
    //freopen("in.txt", "r", stdin);
    //freopen("out.txt", "w", stdout);

    int a, b;
    fun();
    map<int, int>mp;
    
    while(~scanf("%d", &n) && n){
        for(ll i = 0; i <= 300000; i++) f[i] = i;
        for(int i = 0; i < n; i++){
            scanf("%d%d", &a, &b);
            int fi = fid(i);
            int fa = fid(a);
            if (fi != fa){
                f[fa] = fi;
            }
            int fb = fid(b);
            if (fi != fb) f[fb] = fi;
        }
         
        mp.clear();
        for(int i = 0; i < n; i++){
            f[i] = fid(i);   // 补充路径压缩 !!
            mp[f[i]]++;
        }
         
        int cnt = mp.size();
        map<int, int>::iterator it;
        
        ll ans = jc[cnt-1]*fang[cnt-1]%mod;
        //ll ans = 1;  // 这样写就有问题了 !!! 虽然不知道为什么 
        for(it = mp.begin(); it != mp.end(); it++){
            int t = it->second; 
            ans *= 1ll*t*(t-1)/2%mod;
            ans %= mod;
        }
        //ans *= jc[cnt-1]%mod;  
        //ans *= fang[cnt-1]%mod;
        //ans %= mod;
        if (cnt == 1) printf("0\n");
        else
            printf("%lld\n", ans%mod);
    }
    
    return 0;
}

 

2017.12.23 浙江工大 迎新赛

标签:大神   联通   输入   分析   1.0   输出   efi   表示   故障   

原文地址:http://www.cnblogs.com/ccut-ry/p/8097453.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!