Cards BZOJ 1004
Cards
【问题描述】
小春现在很清闲,面对书桌上的N张牌,他决定给每张染色,目前小春只有3种颜色:红色,蓝色,绿色.他询问Sun有多少种染色方案,Sun很快就给出了答案.进一步,小春要求染出Sr张红色,Sb张蓝色,Sg张绝色.他又询问有多少种方案,Sun想了一下,又给出了正确答案. 最后小春发明了M种不同的洗牌法,这里他又问Sun有多少种不同的染色方案.两种染色方法相同当且仅当其中一种可以通过任意的洗牌法(即可以使用多种洗牌法,而每种方法可以使用多次)洗成另一种.Sun发现这个问题有点难度,决定交给你,答案可能很大,只要求出答案除以P的余数(P为质数).
【输入格式】
第一行输入 5 个整数:Sr,Sb,Sg,m,p(m<=60,m+1<p<100)。n=Sr+Sb+Sg。接下来 m 行,每行描述一种洗牌法,每行有 n 个用空格隔开的整数 X1X2...Xn,恰为 1 到 n 的一个排列,表示使用这种洗牌法,第 i位变为原来的 Xi位的牌。输入数据保证任意多次洗牌都可用这 m种洗牌法中的一种代替,且对每种洗牌法,都存在一种洗牌法使得能回到原状态。
【输出格式】
不同染法除以P的余数
【样例输入】
1 1 1 2 7
2 3 1
3 1 2
【样例输出】
2
【样例解释】
有2种本质上不同的染色法RGB和RBG,使用洗牌法231一次可得GBR 和BGR,使用洗牌法312一次可得BRG和GRB。
【数据范围】
100%数据满足:Max{Sr,Sb,Sg}<=20。
题解:
Burnside引理:用D(ai)表示在置换ai下不变的元素个数。L表示本质不同的方案数。G表示置换群。|G|表示置换群的大小。
置换群简单地讲就是给定置换,所有置换的置换所组成的集合就是置换群

举个例子:
4个置换组成的置换群:
1 2 3 4
2 3 4 1
3 4 1 2
4 1 2 3
假设我们有2种颜色用0与1表示
第一种:所有方案都不变,D(a1) = 16
第二种:0000、1111,D(a2) = 2
第三种:0000、1111、0101、1010,D(a3) = 4
第四种:0000、1111,D(a4) = 2
那么 L = (16 + 2 + 4 + 2) / 4 = 6
这就是本质不同的方案数
题目中已经保证给出所有置换中,只要加上一个不变的置换就能得到置换群
显然置换可以拆成多个轮换相乘
所以对每种轮换内的元素进行同种颜色的染色,那么轮换中的颜色经过此置换仍然不变
那么用一个01背包,就可以求出在一个置换下不变的元素个数
然后有模数,所以用逆元求一下分母
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#include<cmath>
using namespace std;
inline int Get()
{
int x = ;
char c = getchar();
while('' > c || c > '') c = getchar();
while('' <= c && c <= '')
{
x = (x << ) + (x << ) + c - '';
c = getchar();
}
return x;
}
const int me = ;
int r, b, g, m, p, n, t;
bool vis[me];
int a[me];
int si[me];
int num;
long long ans;
long long f[me][me][me];
inline long long Pow(long long x, int y)
{
long long res = ;
long long sum = x % p;
while(y)
{
if(y & ) res = (res * sum) % p;
sum = (sum * sum) % p;
y >>= ;
}
return res;
}
int main()
{
r = Get(), b = Get(), g = Get(), m = Get(), p = Get();
n = r + b + g;
++m;
t = m;
while(m--)
{
num = ;
memset(f, , sizeof(f));
memset(si, , sizeof(si));
memset(vis, false, sizeof(vis));
for(int i = ; i <= n; ++i)
{
if(m) a[i] = Get();
else a[i] = i;
}
for(int i = ; i <= n; ++i)
{
if(!vis[i])
{
int c = a[i];
++num;
while(!vis[c])
{
vis[c] = true;
++si[num];
c = a[c];
}
}
}
f[][][] = ;
for(int i = ; i <= num; ++i)
for(int j = r; j >= ; --j)
for(int k = b; k >= ; --k)
for(int l = g; l >= ; --l)
{
if(j >= si[i]) f[j][k][l] = (f[j][k][l] + f[j - si[i]][k][l]) % p;
if(k >= si[i]) f[j][k][l] = (f[j][k][l] + f[j][k - si[i]][l]) % p;
if(l >= si[i]) f[j][k][l] = (f[j][k][l] + f[j][k][l - si[i]]) % p;
}
ans = (ans + f[r][b][g]) % p;
}
printf("%lld", (ans * Pow(t, p - )) % p);
}
Cards BZOJ 1004的更多相关文章
- 【HNOI2008】Cards BZOJ 1004
Description 小春现在很清闲,面对书桌上的N张牌,他决定给每张染色,目 前小春只有3种颜色:红色,蓝色,绿色.他询问Sun有多少种染色方案,Sun很快就给出了答案.进一步,小春要求染出Sr张 ...
- [BZOJ 1004] [HNOI2008] Cards 【Burnside引理 + DP】
题目链接:BZOJ - 1004 题目分析 首先,几个定义和定理引理: 群:G是一个集合,*是定义在这个集合上的一个运算. 如果满足以下性质,那么(G, *)是一个群. 1)封闭性,对于任意 a, b ...
- 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:/ ...
- bzoj 1004 Cards
1004: [HNOI2008]Cards Description 小春现在很清闲,面对书桌上的N张牌,他决定给每张染色,目前小春只有3种颜色:红色,蓝色,绿色.他询问Sun有 多少种染色方案,Sun ...
- bzoj 1004 1004: [HNOI2008]Cards burnside定理
1004: [HNOI2008]Cards Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1668 Solved: 978[Submit][Stat ...
- BZOJ 1004: [HNOI2008]Cards( 置换群 + burnside引理 + 背包dp + 乘法逆元 )
题意保证了是一个置换群. 根据burnside引理, 答案为Σc(f) / (M+1). c(f)表示置换f的不动点数, 而题目限制了颜色的数量, 所以还得满足题目, 用背包dp来计算.dp(x,i, ...
- 【BZOJ 1004】 1004: [HNOI2008]Cards (置换、burnside引理)
1004: [HNOI2008]Cards Description 小春现在很清闲,面对书桌上的N张牌,他决定给每张染色,目前小春只有3种颜色:红色,蓝色,绿色.他询问Sun有多少种染色方案,Sun很 ...
- BZOJ 1004 【HNOI2008】 Cards
题目链接:Cards 听说这道题是染色问题的入门题,于是就去学了一下\(Bunside\)引理和\(P\acute{o}lya\)定理(其实还是没有懂),回来写这道题. 由于题目中保证"任意 ...
- BZOJ 1004: [HNOI2008]Cards
Description 给你一个序列,和m种可以使用多次的置换,用3种颜色染色,求方案数%p. Sol Burnside定理+背包. Burnside定理 \(N(G,\mathbb{C})=\fra ...
随机推荐
- python中的循环语句-01
接触python已经一周时间,下面针对这一周python学习的情况做一梳理: 1)单行注释,使用一个#:多行注释,使用三个单引号(或者三个双引号)如: name = input("name: ...
- Codeforces Round #272 (Div. 2)-A. Dreamoon and Stairs
http://codeforces.com/contest/476/problem/A A. Dreamoon and Stairs time limit per test 1 second memo ...
- LeetCode || 大杂烩w
454. 4Sum II 题意:给四个数组,每个数组内取一个数使得四个数和为0,问有多少种取法 思路:枚举为On4,考虑两个数组,On2枚举所有可能的和,将和的出现次数存入map中,On2枚举另两个数 ...
- 在tomcat中配置连接池
在tomcat的conf/Catalina/localhost目录下配置项目路径,tomcat启动是会直接根据配置去加载项目. 虽然配置就一句话,但经常忘,今天记下来. 如果你的项目成名是:mypro ...
- ios坐标系统
在写程序的时候发现,iOS下的坐标.位置很容易弄乱,特别是在不同的坐标系统中,必须完成弄明白一些概念才能做相应的变化,例如CoreImage和UIView的坐标系统就截然不同,一个是以屏幕的左上角为原 ...
- SVN的配置
Xcode 是开发人员建立 Mac OS X 应用程序的最快捷方式,也是利用新的苹果电脑公司技术的最简单的途径,而SVN是版本控制工具,那么Xcode SVN又是什么呢?如何配置Xcode SVN? ...
- Vue开发微信公众号默认背景为灰色
最近公司有一个项目,使用Vue开发微信公众号,开发过程遇到一个问题,即设计图的整体背景是白色的,但是公众号里默认的背景是浅灰色,如果某个页面高度没能占满一屏,就会露出浅灰色的默认背景,会显得很不协调. ...
- json 将key值以字符串形式取出
int GetJsonCString(const Json::Value& value, char* str, int n){ if (!value.empty() && va ...
- Windows 10 Mac 为Vs Code配置C/C++环境
2019-06-10 更新: 加上Mac版本的Vscode配置文件 0.前言 实现效果:右键一键编译运行C/C++文件 Vs code的代码效果很好看,也很轻量,所以想为Vs Code配置C/C++环 ...
- UITextView 实现placeholder
1.在创建textView的时候,赋值其文本属性 即 textView.text = @"内容": 2.在开始编辑的代理方法中进行如下操作 - (void)textViewDidB ...