[BZOJ 1004][HNOI2008]Cards(Polya定理/Burnside引理)

Description

小春现在很清闲,面对书桌上的N张牌,他决定给每张染色,目前小春只有3种颜色:红色,蓝色,绿色.他询问Sun有
多少种染色方案,Sun很快就给出了答案.进一步,小春要求染出Sr张红色,Sb张蓝色,Sg张绝色.他又询问有多少种方
案,Sun想了一下,又给出了正确答案. 最后小春发明了M种不同的洗牌法,这里他又问Sun有多少种不同的染色方案.
两种染色方法相同当且仅当其中一种可以通过任意的洗牌法(即可以使用多种洗牌法,而每种方法可以使用多次)洗
成另一种.Sun发现这个问题有点难度,决定交给你,答案可能很大,只要求出答案除以P的余数(P为质数)

Solution

Polya学习了一个

用背包来求每种置换的不动点数

(一开始忘记加不变的那种置换了…)

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
using namespace std;
int n,sr,sb,sg,m,p,a[65][65],b[65],f[25][25][25],ans=0;
bool vis[65];
void exgcd(int a,int b,int& d,int& x,int& y)
{
    if(!b){d=a,x=1,y=0;return;}
    exgcd(b,a%b,d,y,x);y-=x*(a/b);
}
int inv(int a,int p)
{
    int d,x,y;
    exgcd(a,p,d,x,y);
    return (x+p)%p;
}
int dp(int cnt)
{
    memset(f,0,sizeof(f));
    f[0][0][0]=1;
    for(int i=1;i<=cnt;i++)
    for(int j=sr;j>=0;j--)
    for(int k=sb;k>=0;k--)
    for(int l=sg;l>=0;l--)
    {
        f[j][k][l]=0;
        if(j>=b[i])f[j][k][l]=(f[j][k][l]+f[j-b[i]][k][l])%p;
        if(k>=b[i])f[j][k][l]=(f[j][k][l]+f[j][k-b[i]][l])%p;
        if(l>=b[i])f[j][k][l]=(f[j][k][l]+f[j][k][l-b[i]])%p;
    }
    return f[sr][sb][sg];
}
int main()
{
    scanf("%d%d%d%d%d",&sr,&sb,&sg,&m,&p);
    n=sr+sb+sg;
    for(int i=1;i<=m+1;i++)
    {
        if(i!=m+1)
        for(int j=1;j<=n;j++)
        scanf("%d",&a[i][j]);
        else{
              for(int j=1;j<=n;j++)
            a[i][j]=j;
        }
        memset(vis,0,sizeof(vis));
        int cnt=0;
        for(int j=1;j<=n;j++)
        {
            if(!vis[j])
            {
                vis[j]=1;b[++cnt]=1;
                int t=a[i][j];
                while(t!=j)
                {vis[t]=1;b[cnt]++;t=a[i][t];}
            }
        }
        ans=(ans+dp(cnt))%p;
    }
    ans=(ans*inv(m+1,p))%p;
    printf("%d\n",ans);
    return 0;
}

 

posted @ 2017-04-14 22:47  Zars19  阅读(229)  评论(0编辑  收藏  举报