Description

给你一个序列,和m种可以使用多次的置换,用3种颜色染色,求方案数%p.

Sol

Burnside定理+背包.

Burnside定理 \(N(G,\mathbb{C})=\frac {1}{\left | G \right |}\sum_{f\in G}\left |\mathbb{C}(f)  \right |\)

\(\mathbb{C}\) 中非等价的着色数等于在 \(G\) 中的置换作用下保持不变的着色的平均数.《组合数学》

对于每一种置换 求出关于置换的一个有向圈(相当于一个连通块),这里有更好的算法,但是我没怎么想,反正范围小直接用的并查集暴力置换合并.

想让计算不定置换,就要将每个有向圈染成同样的颜色,然后用背包求出来.

注意,它可以不用任何一种置换,所以要加上单位元的置换,这个可以直接用组合数学求出来,逆元随便搞搞就可以了.

Code

/**************************************************************
Problem: 1004
User: BeiYu
Language: C++
Result: Accepted
Time:260 ms
Memory:2368 kb
****************************************************************/ #include<cstdio>
#include<cstring>
#include<iostream>
using namespace std; const int N = 65; int n,m,sr,sb,sg,Mo,cnt,ans;
int a[N],tmp[N],sz[N],id[N];
int f[N][N][N],p[N]; inline int in(int x=0,char ch=getchar()){ while(ch>'9'||ch<'0') ch=getchar();
while(ch>='0'&&ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar();return x; }
inline int Pow(int a,int b,int res=1){ for(;b;b>>=1,a=a*a%Mo) if(b&1) res=res*a%Mo;return res; } int Find(int x){ return p[x]==x?x:p[x]=Find(p[x]); }
void Merge(int u,int v){
int f1=Find(u),f2=Find(v);
if(f1!=f2) p[f2]=f1;
}
int work(){
for(int i=1;i<=n;i++) p[i]=i;
for(int i=1;i<=n;i++) a[i]=in();
for(int i=1;i<=n;i++){
for(int i=1;i<=n;i++) Merge(i,a[i]);
for(int i=1;i<=n;i++) tmp[i]=a[a[i]];
for(int i=1;i<=n;i++) a[i]=tmp[i];
// for(int i=1;i<=n;i++) cout<<a[i]<<" ";cout<<endl;
}
cnt=0;
memset(sz,0,sizeof(sz));memset(f,0,sizeof(f));
for(int i=1;i<=n;i++) if(p[i]==i) id[i]=++cnt;
for(int i=1;i<=n;i++) sz[id[Find(i)]]++;
f[0][0][0]=1;
for(int u=1;u<=cnt;u++){
for(int i=sr;~i;i--) for(int j=sb;~j;j--) for(int k=sg;~k;k--){
if(i>=sz[u]) f[i][j][k]=(f[i][j][k]+f[i-sz[u]][j][k])%Mo;
if(j>=sz[u]) f[i][j][k]=(f[i][j][k]+f[i][j-sz[u]][k])%Mo;
if(k>=sz[u]) f[i][j][k]=(f[i][j][k]+f[i][j][k-sz[u]])%Mo;
}
}return f[sr][sb][sg];
}
int main(){
// freopen("in.in","r",stdin);
sr=in(),sb=in(),sg=in(),m=in(),Mo=in(),n=sr+sb+sg; ans=1;
for(int i=1;i<=n;i++) ans=(ans*i)%Mo;
for(int i=1;i<=sr;i++) ans=(ans*Pow(i,Mo-2))%Mo;
for(int i=1;i<=sb;i++) ans=(ans*Pow(i,Mo-2))%Mo;
for(int i=1;i<=sg;i++) ans=(ans*Pow(i,Mo-2))%Mo; for(int i=1;i<=m;i++) ans=(ans+work())%Mo;
cout<<ans*Pow(m+1,Mo-2)%Mo<<endl;
return 0;
}

  

BZOJ 1004: [HNOI2008]Cards的更多相关文章

  1. BZOJ 1004: [HNOI2008]Cards( 置换群 + burnside引理 + 背包dp + 乘法逆元 )

    题意保证了是一个置换群. 根据burnside引理, 答案为Σc(f) / (M+1). c(f)表示置换f的不动点数, 而题目限制了颜色的数量, 所以还得满足题目, 用背包dp来计算.dp(x,i, ...

  2. [BZOJ 1004] [HNOI2008] Cards 【Burnside引理 + DP】

    题目链接:BZOJ - 1004 题目分析 首先,几个定义和定理引理: 群:G是一个集合,*是定义在这个集合上的一个运算. 如果满足以下性质,那么(G, *)是一个群. 1)封闭性,对于任意 a, b ...

  3. bzoj 1004 [HNOI2008]Cards && poj 2409 Let it Bead ——置换群

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1004 http://poj.org/problem?id=2409 学习材料:https:/ ...

  4. BZOJ 1004 HNOI2008 Cards Burnside引理

    标题效果:特定n张卡m换人,编号寻求等价类 数据保证这m换人加上置换群置换后本身构成 BZOJ坑爹0.0 条件不那么重要出来尼玛怎么做 Burnside引理--昨晚为了做这题硬啃了一晚上白书0.0 都 ...

  5. BZOJ 1004: [HNOI2008]Cards(群论)

    好吧我就是蒟蒻根本没听说过群论(虽说听叉姐说几万年都不会考) 我也讲不太来,直接戳VFK大神的blog啦 = = http://vfleaking.blog.163.com/blog/static/1 ...

  6. BZOJ 1004: [HNOI2008]Cards [Polya 生成函数DP]

    传送门 题意:三种颜色,规定使用每种颜色次数$r,g,b$,给出一个置换群,求多少种不等价着色 $m \le 60,\ r,g,b \le 20$ 咦,规定次数? <组合数学>上不是有生成 ...

  7. bzoj 1004 1004: [HNOI2008]Cards burnside定理

    1004: [HNOI2008]Cards Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1668  Solved: 978[Submit][Stat ...

  8. 【BZOJ 1004】 1004: [HNOI2008]Cards (置换、burnside引理)

    1004: [HNOI2008]Cards Description 小春现在很清闲,面对书桌上的N张牌,他决定给每张染色,目前小春只有3种颜色:红色,蓝色,绿色.他询问Sun有多少种染色方案,Sun很 ...

  9. 【BZOJ】1004: [HNOI2008]Cards(置换群+polya+burnside)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1004 学习了下polya计数和burnside引理,最好的资料就是:<Pólya 计数法的应用 ...

随机推荐

  1. easyUI中tree的简单使用

    一.在JS中的代码 $('#tt').tree({ url: baseCtx + 'lib/easyui-1.4/demo/tree/tree_data1.json',//tree数据的来源,json ...

  2. Java关键字——throws和throw

    throws关键字 在定义一个方法时,可以使用throws关键字声明,使用throws声明的方法表示此方法不处理异常,而交给方法的调用处进行处理. 使用了throws关键字,表示不管是否会有异常,在调 ...

  3. 端口扫描(TCP)

    还待优化... #include <string.h> #include <WinSock.h> #include <stdio.h> #pragma commen ...

  4. yum安装指定(特定)版本(旧版本)软件包的方法

    在命令行里输入: yum list SDL 注意这里类库的名字是区别大小写的. 参考 http://www.dabu.info/yum-install-specific-version-old-pac ...

  5. Debian 8安装ibus输入法

    # apt-get install ibus ibus-sunpinyin ibus-table-wubi

  6. Index/Common目录下文件

    1.在Common目录下创建Common.php(系统会自动加载Common.php) 代码: function say(){ echo '; } 在IndecAction.php输出 public ...

  7. ecshop怎么添加配送方式

    步骤1.打开includes\modules\shipping文件夹,把sto_express.php复制多一份,重名为tt_express.php: 步骤2.打开tt_express.php,ctr ...

  8. Java常见的几种内存溢出及解决方法

    Java常见的几种内存溢出及解决方法[情况一]:java.lang.OutOfMemoryError:Javaheapspace:这种是java堆内存不够,一个原因是真不够(如递归的层数太多等),另一 ...

  9. myeclipse工程当中的.classpath 和.project文件什么作用?

    .project是项目文件,项目的结构都在其中定义,比如lib的位置,src的位置,classes的位置.classpath的位置定义了你这个项目在编译时所使用的$CLASSPATH .classpa ...

  10. 开源项目管理平台*redmine*的架设

    yum -y install ruby yum install rubygems gem install heroku gem install rack -v=1.0.1 gem install ru ...