埋锅。。。BZOJ1004-置换群+burnside定理+
看这道题时当时觉得懵逼。。。这玩意完全看不懂啊。。。什么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定理+的更多相关文章
- BZOJ1004 [HNOI2008]Cards 【burnside定理 + 01背包】
题目链接 BZOJ1004 题解 burnside定理 在\(m\)个置换下本质不同的染色方案数,等于每种置换下不变的方案数的平均数 记\(L\)为本质不同的染色方案数,\(m\)为置换数,\(f(i ...
- 我对Burnside定理的理解
我想了想,发现可以证明burnside定理. 置换:n个元素1,2,-,n之间的一个置换表示1被1到n中的某个数a1取代,2被1到n中的某个数a2取代,直到n被1到n中的某个数an取代,且a1,a2, ...
- poj 2409+2154+2888(Burnside定理)
三道burnside入门题: Burnside定理主要理解置换群置换后每种不动点的个数,然后n种不动点的染色数总和/n为answer. 对于旋转,旋转i个时不动点为gcd(n,i). 传送门:poj ...
- HUST 1569(Burnside定理+容斥+数位dp+矩阵快速幂)
传送门:Gift 题意:由n(n<=1e9)个珍珠构成的项链,珍珠包含幸运数字(有且仅由4或7组成),取区间[L,R]内的数字,相邻的数字不能相同,且旋转得到的相同的数列为一种,为最终能构成多少 ...
- 【Burnside定理】&【Pólya定理】
Burnside & Pólya (详细内容请参阅<组合数学>或2008年cyx的论文,这里只写一些我学习的时候理解困难的几个点,觉得我SB的请轻鄙视……如果有觉得不科学的地方欢迎 ...
- poj2409:Let it Bead(置换群 polya定理)
题目大意:长度为n的项链,要染m种颜色,可以通过旋转或翻转到达的状态视为同一种,问有多少种染色方案. 学了一波polya定理,发现很好理解啊,其实就是burnside定理的扩展. burnside定理 ...
- 置换群 Burnside引理 Pólya定理(Polya)
置换群 设\(N\)表示组合方案集合.如用两种颜色染四个格子,则\(N=\{\{0,0,0,0\},\{0,0,0,1\},\{0,0,1,0\},...,\{1,1,1,1\}\}\),\(|N|= ...
- [bzoj1004][HNOI2008][Cards] (置换群+Burnside引理+动态规划)
Description 小春现在很清闲,面对书桌上的N张牌,他决定给每张染色,目前小春只有3种颜色:红色,蓝色,绿色.他询问Sun有多少种染色方案,Sun很快就给出了答案.进一步,小春要求染出Sr张红 ...
- bzoj1004 [HNOI2008]Cards Burnside定理+背包
题目传送门 思路:首先是Burnside引理,要先学会这个博客. Burnside引理我们总结一下,就是 每种置换下不动点的数量之和除以置换的总数,得到染色方案的数量. 这道题,显然每种 ...
随机推荐
- 2018(2017)美图java服务端笔试(回忆录)
选择题有几道,是比较基础的 填空题两道:一道是类似c语言的给出abc的值求 ++a+b+++c++ ,另一道是说出两个常见的垃圾回收算法 编程题 找出出现次数为1的数字然后改进(要求O(n)) 数据 ...
- maven与jdk版本对应关系
Maven发布历史 发布日期 版 必需的Java版本 链接 2018年6月21日 3.5.4 Java 7 宣布,发布说明,参考文档 2018年3月8日 3.5.3 宣布,发布说明,参考文档 2017 ...
- logstash关于date时间处理的几种方式总结
1.第一种,直接在配置文件中自定义时间格式 这是tomcat配置文件中的一段日志时间配置,按照这样的配置,那么输出的日志是这样子的: 然后你继续在logstash中这样子配置 此时logstash就不 ...
- php学习----面向对象
面向对象 项目代码都应该由单个能起子程序作用的对象组成 重用性.灵活性.扩展性 变量 $this 代表自身的对象. PHP_EOL 为换行符. PHP 中创建对象 类创建后,我们可以使用 new 运算 ...
- 【2018.05.09 Python学习及实践】个人项目中使用的Python库备忘-持续更新
科研中无论是使用C/C++.Python.Matlab,如果能找到合适的库可谓是事半功倍: 有时候忙活半天才发现本身就有成熟的库可用,自己实现的在功能.性能.安全性上都远远不及,虽然锻炼了能力,但存在 ...
- 【微信小游戏】【提审的坑】!#¥%&……&&……%¥#@@*()()&%%¥
一.开通了虚拟支付后审核变慢 前两个版本是没有开通虚拟支付的,最快一个半小时就过审了.当时还在暗自嘲笑WX,条款很多,审核却那么松,甚至一度怀疑是不是没有审核直接放.然而第三版提审了之后,一个小时.两 ...
- Myeclipse2017C版本破解
原文地址: https://blog.csdn.net/qq_33945246/article/details/79589152
- C. Playlist Educational Codeforces Round 62 (Rated for Div. 2) 贪心+优先队列
C. Playlist time limit per test 2 seconds memory limit per test 256 megabytes input standard input o ...
- 设计模式のFilterPattern(过滤器模式)----结构模式
一.产生背景 我们有一堆“人”的对象,我们应该怎么选择出其中的男性.女性或者其他类型的呢?这时候我们可以用过滤器模式 二.通常做法 我们将创建一个 Person 对象.Criteria 接口和实现了该 ...
- EJB3.0之事务
版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/aboy123/article/details/24586803 一 事务是什么 1 事务的概念 1. ...