zoukankan      html  css  js  c++  java
  • bzoj1004 [HNOI2008]Cards Burnside定理+背包

    题目传送门

    思路:首先是Burnside引理,要先学会这个博客。

              Burnside引理我们总结一下,就是 每种置换下不动点的数量之和除以置换的总数,得到染色方案的数量。

            这道题,显然每种洗牌方式都是一种置换,我们先数出每种置换的不动点。什么叫不动点,就是在这个置换下不停的变化后状态不变的染色方案。容易想出每个置换都有一个循环节,每张牌在某种洗牌方式下的位置是循环的,那要使得这个成为一个不动点,就需要使得同一循环节上的牌的颜色相同。那么这个问题就转化成了一个三维背包问题了。

      背包的转移方程为$f[i][j][k]+=f[i-size][j][k]+f[i][j-size][k]+f[i][j][k-size]$。

      接下来就是一个简单的逆元了,注意本身不动也是一种染色方案。

    #include<bits/stdc++.h>
    #define clr(a,b) memset(a,b,sizeof(a))
    #define fpn() freopen("simple.in","r",stdin)
    #define rd read()
    using namespace std;
    typedef long long ll;
    inline int read()
    {
        int x=0,t=1;char ch=getchar();
        while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
        if(ch=='-')t=-1,ch=getchar();
        while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
        return x*t;
    }
    const int maxn=110;
    int p,a[maxn],vis[maxn],f[maxn][maxn][maxn],siz[maxn],tot;
    int sr,sb,sg,m,ans,n;
    int solve(){
        clr(vis,0),clr(f,0),tot=0;
        int len=0;
        for(int i=1;i<=n;i++){
            len=0;
            if(!vis[i]){
                int k=i;
                while(!vis[k]){
                    len++;
                    vis[k]=1;
                    k=a[k];
                }
                siz[++tot]=len;
            }
        }
        f[0][0][0]=1;
        for(int s=1;s<=tot;s++)
        {
            for(int i=sr;i>=0;i--)
            {
                for(int j=sb;j>=0;j--)
                {
                    for(int k=sg;k>=0;k--)
                    {
                        if(i>=siz[s])f[i][j][k]=(f[i][j][k]+f[i-siz[s]][j][k])%p;
                        if(j>=siz[s])f[i][j][k]=(f[i][j][k]+f[i][j-siz[s]][k])%p;
                        if(k>=siz[s])f[i][j][k]=(f[i][j][k]+f[i][j][k-siz[s]])%p;
                    }
                }
            }
        }
        return f[sr][sb][sg];
        
    }
    int qpow(int a,int b){
        int res=1;
        while(b){
            if(b&1){
                res*=a;
                res%=p;
            }
            b>>=1;
            a*=a;
            a%=p;
        }
        return res;
    }
    int main(){
        cin>>sr>>sb>>sg>>m>>p;
        n=sr+sb+sg;
        for(int i=1;i<=n;i++)a[i]=i;
        int ans=0;
        ans+=solve()%p;
        for(int t=1;t<=m;t++)
        {
            for(int i=1;i<=n;i++)
            {
                scanf("%d",&a[i]);
            }
            ans=(ans+solve())%p;
        }
        ans=ans*qpow(m+1,p-2)%p;
        cout<<ans<<endl;
    }
    View Code
  • 相关阅读:
    如何给远程主机开启mysql远程登录权限
    Session机制详解
    CentOS 下PHP的卸载
    PHP实现执行定时任务的几种思路详解
    容易产生错误的where条件
    php超时任务处理
    (转载)Android项目tab类型主界面总结
    使用xutils发送POST请求,携带json和图片二进制文件数据获取服务器端返回json数据
    Android开发中常见错误
    (转载)Android显示原理简介
  • 原文地址:https://www.cnblogs.com/mountaink/p/10439573.html
Copyright © 2011-2022 走看看