cf859EDesk Disorder

Posted yoyoball

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了cf859EDesk Disorder相关的知识,希望对你有一定的参考价值。

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
",ans);
}

以上是关于cf859EDesk Disorder的主要内容,如果未能解决你的问题,请参考以下文章

CF859C Pie Rules

Codeforces Round 859 (Div. 4) 题解集

第859期成为一名函数式码农之二

如何从后台弹出片段

Codeforces 859C - Pie Rules

859. 亲密字符串