埋锅。。。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引理我们总结一下,就是 每种置换下不动点的数量之和除以置换的总数,得到染色方案的数量. 这道题,显然每种 ...
随机推荐
- WebAPI接口设计:SwaggerUI文档 / 统一响应格式 / 统一异常处理 / 统一权限验证
为什么还要写这类文章?因为我看过网上很多讲解的都不够全面,而本文结合实际工作讲解了swaggerui文档,统一响应格式,异常处理,权限验证等常用模块,并提供一套完善的案例源代码,在实际工作中可直接参考 ...
- Gradle: Gradle Wrapper
[Gradle 安装] 安装完毕后,记得设置一下环境变量.Environment Variables:GRADLE_HOME=D:\Program Files\Gadle\gradle-4.7Path ...
- Syntax error, parameterized types are only available if source level is 1.5 解决方案
在网上找了一个K-means算法的程序,打开,运行,出现了Syntax error,parameterized types are only available if source level is ...
- "敏捷革命"读书笔记
最近看可一本书 书名叫<敏捷革命>外国著作中文翻译 本来想自己总结读后感但是本书后面都有本章的总结,所以下面都已摘抄为主,以备之后快速浏览 第一章 世界的运作方式已经打破 规划是有用的,而 ...
- Unity Shader 基础(1): RenderType & ReplacementShader
很多Shader中都会定义RenderType这个类型,但是一直搞不明白到底是干嘛的,官方文档是这样结解释的:Rendering with Replaced Shaders Rendering wit ...
- js获取请求地址后面带的参数
浏览器输入页面地址的时候在后面带有请求参数, 页面加载后需要获取携带的参数, 可以使用js, 在页面加载js的时候获取参数 http://localhost:8080/demo/index.html? ...
- MacBook Pro维修过程
上个月买了2016款无touchbar的MacBook Pro,这款的键盘跟15年版本的很不一样. 在用了三周左右,就发现有个键按着不灵活了,像是进了水一样,黏住感觉,必须要大力按才会出来.思来想去, ...
- spring简述
1. 什么是Spring Spring:SE/EE开发的一站式框架.(有EE开发每一层的解决方案) WEB层:SpringMVC Service层:Spring的Bean管理,Spring的声明式事务 ...
- Pandas 处理丢失数据
处理丢失数据 import pandas as pd from pandas import Series, DataFrame import numpy as np 有两种丢失数据: 1. None ...
- Springboot 实现api校验和登录验证
https://blog.csdn.net/qq_36085004/article/details/83348144 文章目录 API校验 场景 实现思路 代码 拦截器: 拦截器注册: 登录token ...