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

【cf859E】Desk Disorder

时间:2018-10-28 11:55:27      阅读:141      评论:0      收藏:0      [点我收藏+]

标签:code   bug   can   ace   using   lin   ios   space   open   

Portal --> cf859E

Solution

??  我们可以将每一个人看成一条边,将位置看成点,然后一个人在新的方案中可以选择的位置就是这条边连接的两个点,然后我们就得到了一个图

?  注意到这个图可能包含多个连通块,每个连通块可以独立计算,那么最后的答案应该就是各个连通块计算结果的乘积,那么现在我们的问题就是怎么算一个连通块内的答案

??  考虑一个包含\(n\)个点的连通块,我们来分析一下这个连通块的性质:首先因为这是一个连通块,所以边数\(m>=n-1\),然后注意到这题中有一个性质:原方案中没有两个人选的是同一个位置(点),所以我们可以得到边数\(m<=n\)

?  所以一个包含\(n\)个点的连通块只可能有两种情况:

1、\(m=n-1\):这个时候是一棵树,那么答案就应该是\(n\)(树中的节点个数),具体的话就是考虑哪个节点没有人选,所以就有\(n\)种方案

2、\(m=n\):这个时候如果是一个基环树,考虑环中的边数和点数相同,所以环中所有的点只能被环上的边选,这样一来树中能选的点数和要选的边数相等,所以树中的方案是唯一的,环中的方案只有两种,所以这个时候有\(2\)中方案;但是!如果说这个环是个自环,那就只有一种方案了,因为环中不能移动

?  所以我们直接并查集判断一下统计一下就好了

?  

??  代码大概长这个样子

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int N=1e5+10,MOD=1e9+7;
int f[N*2],in[N*2],cnt[N*2],mark[N*2];
int n,m,tot,ans;
int get_f(int x){return f[x]=f[x]==x?f[x]:get_f(f[x]);}
int mul(int x,int y){return 1LL*x*y%MOD;}

int main(){
#ifndef ONLINE_JUDGE
    freopen("a.in","r",stdin);
#endif
    int x,y;
    scanf("%d",&n);
    for (int i=1;i<=2*n;++i) f[i]=i,cnt[i]=1,mark[i]=0;
    ans=1;
    int tmp=0;
    for (int i=1;i<=n;++i){
        scanf("%d%d",&x,&y); 
        get_f(x); get_f(y);
        mark[x]=1;
        tmp=max(tmp,max(x,y));
        if (x==y) continue;
        if (f[x]==f[y]) ans=mul(ans,2);
        cnt[f[y]]+=cnt[f[x]];
        f[f[x]]=f[y];
    }
    int debug=0;
    for (int i=1;i<=2*n;++i)
        if (!mark[i])
            ans=mul(ans,cnt[i]),++debug;
    printf("%d\n",ans);
}

【cf859E】Desk Disorder

标签:code   bug   can   ace   using   lin   ios   space   open   

原文地址:https://www.cnblogs.com/yoyoball/p/9543299.html

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