看这道题时当时觉得懵逼。。。这玩意完全看不懂啊。。。什么burnside。。。难受。。。

于是去看了点视频和资料,大概懂了置换群和burnside定理,亦步亦趋的懂了别人的代码,然后慢慢的打了出来。。。高兴的一匹。

回归正题啊,这个题如果大家不懂置换群的概念。。。是很难看的懂的,M种洗牌,代表了M种置换,加上自己本身一种,构成看M+1种置换,如果一种可以通过任意的洗牌法洗成另一种的看成一类(这就是等价类的定义),问有多少种染色方法???

这道题很明显嘛,就是算等价类的数目,这个需要用到burnside定理。。。

我们可以求所有循环长度为1的循环的个数,用burnside来求

对于置换i,求出它的循环节,每个循环节必须染成同一个颜色,但是对总数有限制

f[i][j][k]表示用i个红色,j个蓝色,k个绿色的方案数

把每个循环节视为一个物品,权值为长度。

f[i][j][k]=f[i-d[p]][j][k]+f[i][j-d[p]][k]+f[i][j][k-d[p]]

d[p]表示第p个循环节的长度

注意枚举顺序,p要在最外面,保证每个物品只取一次

置换的循环在不变元素中一定是一个颜色

嘤嘤。。。。我也没有懂DP啊。如果哪个大佬懂了,一定要给我说一下啊!!!


啊我懂了,这个dp[i]就代表这个循环的长度,那么我们f[i][j][k]=f[i-d[p]][j][k]+f[i][j-d[p]][k]+f[i][j][k-d[p]] 就代表这个循环分别全部是是红,蓝,绿的数目之和,求到最后的f[sr][sb][sg]就是全部的循环长度为1的个数。

代码

#include<iostream>
#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<map>
using namespace std;
int sr,sb,sg,m,p;
int a[];//记录置换
int b[];//b[i]表示循环长度为i
int f[][][];//f[i][j][k]在i个红色,j个黄色.k个蓝色
int vis[];//记录是否是否在一个类里面
int n;
int qpow(int a,int b){
int ans=;
while(b){
if (b&)ans=ans*a%p;
a=a*a%p;
b/=;
}
return ans%p;
}
int dfs(){
memset(f,,sizeof(f));
memset(vis,,sizeof(vis));
int tot=;
for (int i=;i<=n;i++){//求循环的个数和阶数
if (!vis[i]){
vis[i]=;
tot++;
b[tot]=;
int j=i;
while(!vis[a[j]]){
j=a[j];
vis[j]=;
b[tot]++;
}
}
}
f[][][]=;//求所有循环长度为1的循环的个数 置换的循环在不变元素中一定是一个颜色
for (int z=;z<=tot;z++){
for (int i=sr;i>=;i--){
for (int j=sb;j>=;j--){
for (int k=sg;k>=;k--){
if (i>=b[z])f[i][j][k]+=f[i-b[z]][j][k];
if (j>=b[z])f[i][j][k]+=f[i][j-b[z]][k];
if (k>=b[z])f[i][j][k]+=f[i][j][k-b[z]];
f[i][j][k]=f[i][j][k]%p;
}
}
}
}
return f[sr][sb][sg];
}
int main(){
int ans;
while(~scanf("%d%d%d%d%d",&sr,&sb,&sg,&m,&p)){
n=sr+sb+sg;
ans=;
for(int i=;i<=m;i++){
for (int j=;j<=n;j++){
scanf("%d",&a[j]);
}
ans+=dfs();
ans=ans%p;
}
m++;
for (int i=;i<=n;i++)a[i]=i;
ans+=dfs();ans=ans%p;
ans=ans*qpow(m,p-)%p;
printf("%d\n",ans);
}
return ;
}

#include<iostream>#include<stdio.h>#include<string.h>#include<algorithm>#include<map>using namespace std;int sr,sb,sg,m,p;int a[100];//记录置换int b[100];//b[i]表示循环长度为iint f[30][30][30];//f[i][j][k]在i个红色,j个黄色.k个蓝色int vis[100];//记录是否是否在一个类里面int n;int qpow(int a,int b){  int ans=1;  while(b){    if (b&1)ans=ans*a%p;    a=a*a%p;    b/=2;  }  return ans%p;}int dfs(){  memset(f,0,sizeof(f));  memset(vis,0,sizeof(vis));  int tot=0;  for (int i=1;i<=n;i++){//求循环的个数和阶数    if (!vis[i]){        vis[i]=1;        tot++;        b[tot]=1;        int j=i;        while(!vis[a[j]]){            j=a[j];            vis[j]=1;            b[tot]++;        }    }  }  f[0][0][0]=1;//求所有循环长度为1的循环的个数 置换的循环在不变元素中一定是一个颜色  for (int z=1;z<=tot;z++){     for (int i=sr;i>=0;i--){        for (int j=sb;j>=0;j--){            for (int k=sg;k>=0;k--){              if (i>=b[z])f[i][j][k]+=f[i-b[z]][j][k];              if (j>=b[z])f[i][j][k]+=f[i][j-b[z]][k];              if (k>=b[z])f[i][j][k]+=f[i][j][k-b[z]];              f[i][j][k]=f[i][j][k]%p;            }        }     }  }   return f[sr][sb][sg];}int main(){  int ans;  while(~scanf("%d%d%d%d%d",&sr,&sb,&sg,&m,&p)){     n=sr+sb+sg;     ans=0;     for(int i=1;i<=m;i++){        for (int j=1;j<=n;j++){           scanf("%d",&a[j]);        }        ans+=dfs();        ans=ans%p;     }    m++;    for (int i=1;i<=n;i++)a[i]=i;    ans+=dfs();ans=ans%p;    ans=ans*qpow(m,p-2)%p;    printf("%d\n",ans);  }  return 0;}

埋锅。。。BZOJ1004-置换群+burnside定理+的更多相关文章

  1. BZOJ1004 [HNOI2008]Cards 【burnside定理 + 01背包】

    题目链接 BZOJ1004 题解 burnside定理 在\(m\)个置换下本质不同的染色方案数,等于每种置换下不变的方案数的平均数 记\(L\)为本质不同的染色方案数,\(m\)为置换数,\(f(i ...

  2. 我对Burnside定理的理解

    我想了想,发现可以证明burnside定理. 置换:n个元素1,2,-,n之间的一个置换表示1被1到n中的某个数a1取代,2被1到n中的某个数a2取代,直到n被1到n中的某个数an取代,且a1,a2, ...

  3. poj 2409+2154+2888(Burnside定理)

    三道burnside入门题: Burnside定理主要理解置换群置换后每种不动点的个数,然后n种不动点的染色数总和/n为answer. 对于旋转,旋转i个时不动点为gcd(n,i). 传送门:poj ...

  4. HUST 1569(Burnside定理+容斥+数位dp+矩阵快速幂)

    传送门:Gift 题意:由n(n<=1e9)个珍珠构成的项链,珍珠包含幸运数字(有且仅由4或7组成),取区间[L,R]内的数字,相邻的数字不能相同,且旋转得到的相同的数列为一种,为最终能构成多少 ...

  5. 【Burnside定理】&【Pólya定理】

    Burnside & Pólya (详细内容请参阅<组合数学>或2008年cyx的论文,这里只写一些我学习的时候理解困难的几个点,觉得我SB的请轻鄙视……如果有觉得不科学的地方欢迎 ...

  6. poj2409:Let it Bead(置换群 polya定理)

    题目大意:长度为n的项链,要染m种颜色,可以通过旋转或翻转到达的状态视为同一种,问有多少种染色方案. 学了一波polya定理,发现很好理解啊,其实就是burnside定理的扩展. burnside定理 ...

  7. 置换群 Burnside引理 Pólya定理(Polya)

    置换群 设\(N\)表示组合方案集合.如用两种颜色染四个格子,则\(N=\{\{0,0,0,0\},\{0,0,0,1\},\{0,0,1,0\},...,\{1,1,1,1\}\}\),\(|N|= ...

  8. [bzoj1004][HNOI2008][Cards] (置换群+Burnside引理+动态规划)

    Description 小春现在很清闲,面对书桌上的N张牌,他决定给每张染色,目前小春只有3种颜色:红色,蓝色,绿色.他询问Sun有多少种染色方案,Sun很快就给出了答案.进一步,小春要求染出Sr张红 ...

  9. bzoj1004 [HNOI2008]Cards Burnside定理+背包

    题目传送门 思路:首先是Burnside引理,要先学会这个博客. Burnside引理我们总结一下,就是 每种置换下不动点的数量之和除以置换的总数,得到染色方案的数量.        这道题,显然每种 ...

随机推荐

  1. MyBatis笔记----(2017年)最新的报错:Cannot find class [org.apache.commons.dbcp.BasicDataSource] for bean with name 'dataSource' defined in class path resource [com/ij34/mybatis/applicationContext.xml]; nested e

    四月 05, 2017 4:56:11 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRef ...

  2. CentOS7中启动Tomcat后,8080端口不能被外部访问的解决办法。

    运行:/sbin/iptables -I INPUT -p tcp --dport 8080 -j ACCEPT

  3. 关于getdate()的不同的日期格式

    在使用Sql Server查询数据库时,我们经常会需要查询日期格式的数据,对于日期在sql语言中的格式有一定的要求,通过修改convert中的最后一位参数,可以返回不通格式的时间,具体实现如下: Se ...

  4. 将web项目部署到阿里云服务器上

    一.用eclipse将写好的项目打成war包 1.右键点击项目,选择Export 选择打包之后的路径,点击Finsh. 2.购买阿里云服务器 按下图选择 3.设置云服务器防火墙 4.远程连接云服务器 ...

  5. AndroidNDK开发中使用CMake编译JNI

    虽然一直在做NDK的开发工作,但是由于项目比较久远,都是使用Makefile进行底层编译,对于目前AndroidStudio官方提供的CMake编译方式并不是很了解,现在学习下这种已经不算新潮的新方式 ...

  6. puppet 横向扩展(三)

    Table of Contents 1. 概述 2. 实验环境 3. 实验步骤 3.1. 机器B 的配置 3.2. 机器A 的配置 3.3. 测试配置结果 概述 横向扩展实验之三 – 将CA 认证服务 ...

  7. 【汤鸿鑫 3D太极】肩与膀的细分

  8. Django-rest-framework 接口实现 rest_framework 中有已经定义好的 工具类 mixins generics viewsets

    rest_framework.mixins 请求业务 的 5 种实现 ​ mixin(混合类):不能单独使用,和其它类搭配起来使用(利用了Python支持多继承) rest_framework.mix ...

  9. Linux 系统故障排查和修复技巧

    Linux 系统故障排查和修复技巧 我发现Linux系统在启动过程中会出现一些故障,导致系统无法正常启动,我在这里写了几个应用单用户模式.GRUB命令操作.Linux救援模式的故障修复案例帮助大家了解 ...

  10. Oracle hint之ORDERED和USE_NL

    Hint:ORDERED和USE_NL ORDERED好理解,就是表示根据 from 后面表的顺序join,从左到右,左边的表做驱动表 use_nl(t1,t2):表示对表t1.t2关联时采用嵌套循环 ...