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