# Description

小春现在很清闲, 面对书桌上的 \(N\) 张牌, 他决定给每张染色, 目前小春只有 \(3\) 种颜色: 红色, 蓝色, 绿色. 他询问 Sun 有

多少种染色方案, Sun 很快就给出了答案. 进一步, 小春要求染出 \(Sr\) 张红色, \(Sb\) 张蓝色, \(Sg\) 张绿色. 他又询问有多少种方

案, Sun 想了一下, 又给出了正确答案. 最后小春发明了 \(M\) 种不同的洗牌法, 这里他又问 Sun 有多少种不同的染色方案.

两种染色方法相同当且仅当其中一种可以通过任意的洗牌法 (即可以使用多种洗牌法, 而每种方法可以使用多次) 洗

成另一种. Sun 发现这个问题有点难度, 决定交给你, 答案可能很大, 只要求出答案除以 \(P\) 的余数 (\(P\) 为质数).

并且数据满足 :

输入数据保证任意多次洗牌都可用这 m 种洗牌法中的一种代替,且对每种洗牌法,都存在一种洗牌法使得能回到原状态。

\(Max{(Sr,Sb, Sg)}<=20\)

Solution

群论中的\(burnside\)定理

参考资料: 《组合数学》 或者 网上博客, 感觉《组合数学》讲的很棒QuQ

首先看\(m\)种洗牌, 任意多次都可以用 一种洗牌法来代替, 满足置换群中 合成运算的封闭性

对每种洗牌法, 都存在一种洗牌法使得回到原状态, 满足置换群中 逆元的封闭性

接下来只需要出现单位元就能让洗牌法构成一个置换群

单位元 : 保持所有位置不变的洗牌法(相当于没洗牌)

如果输入的洗牌法没有单位元, 则手动加上一种洗牌法。 就构成了一个置换群。

然后就可以引用 \(burnside\)定理

设 \(f\) 为一种洗牌法,\(c\) 为一种染色法。

我们可以找到有多少个 \(c\) 使得, \(c\) 通过洗牌法\(f\) 洗牌后不变, 记为\(cnt(f)\)。

则最终答案 :

\[ans= \sum cnt(f) \ \div m
\]

证明略(懒得打噜, 网上有)

接下来就要求出 \(cnt(f)\)了:

对于一种洗牌法, 把每个位置都看作一个点, 洗牌的移动看成一条边。 则每个点的出度和入度都是\(1\)

于是构成了若干个不相交的环。要使洗牌后颜色不变, 那么必有同一个环上的颜色相同。

可以通过简单的\(DP\)求出\(cnt(f)\)了!

总复杂度\(O(MN^3)\) ,BZOJ上跑了\(36ms\)

Code

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#define ll long long
#define rd read()
using namespace std; const int N = 62; int n, m, sa, sb, sc, mod;
int nxt[N][N], vis[N], f[22][22];
vector<int> v[N]; int read() {
int X = 0, p = 1; char c = getchar();
for (; c > '9' || c < '0'; c = getchar())
if (c == '-') p = -1;
for (; c >= '0' && c <= '9'; c = getchar())
X = X * 10 + c - '0';
return X * p;
} int fpow(int a, int b) {
int res = 1;
for (; b; b >>= 1, a = a * a % mod)
if (b & 1) res = res * a % mod;
return res;
} void init(int x) {
memset(vis, 0, sizeof(vis));
for (int i = 1; i <= n; ++i) if (!vis[i]) {
vis[i] = 1;
int now = nxt[x][i], tmp = 1;
while (now != i) vis[now] = 1, tmp++, now = nxt[x][now];
v[x].push_back(tmp);
}
} int cal(int x) {
memset(f, 0, sizeof(f));
f[sa][sb] = 1;
int tot = 0;
for (int i = 0, up = v[x].size(); i < up; ++i) {
for (int j = 0; j <= sa; ++j)
for (int k = 0; k <= sb; ++k) {
if (n - tot - j - k < v[x][i]) f[j][k] = 0;
if (j + v[x][i] <= sa) (f[j][k] += f[j + v[x][i]][k]) %= mod;
if (k + v[x][i] <= sb) (f[j][k] += f[j][k + v[x][i]]) %= mod;
}
tot += v[x][i];
}
return f[0][0];
} int main()
{
sa = rd; sb = rd; sc = rd; m = rd; mod = rd;
n = sa + sb + sc;
bool flag = false;
for (int i = 1; i <= m; ++i) {
bool is = true;
for (int j = 1; j <= n; ++j) {
nxt[i][j] = rd;
if (nxt[i][j] != j) is = false;
}
if (is) flag = true;
}
if (!flag) {
m++;
for (int i = 1; i <= n; ++i)
nxt[m][i] = i;
}
for (int i = 1; i <= m; ++i)
init(i);
int ans = 0;
for (int i = 1; i <= m; ++i)
ans = (ans + cal(i)) % mod;
ans = ans * fpow(m, mod - 2);
ans = (ans + mod) % mod;
printf("%d\n", ans);
}

1004: [HNOI2008]Cards - burnside + DP的更多相关文章

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

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

  2. BZOJ 1004 HNOI2008 Cards Burnside引理

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

  3. 1004: [HNOI2008]Cards burnside定理

    https://www.lydsy.com/JudgeOnline/problem.php?id=1004 输入数据保证任意多次洗牌都可用这 m种洗牌法中的一种代替,且对每种洗牌法,都存在一种洗牌法使 ...

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

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

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

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

  6. [HNOI2008]Cards(dp,Burnside引理)

    Burnside引理: 参考自 某大佬对Burnside引理和Polya定理的讲解 相关概念 群:在数学中,群表示一个拥有满足封闭性.满足结合律.有单位元.有逆元的二元运算的代数结构. 置换群:由有限 ...

  7. 【BZOJ1004】[HNOI2008]Cards Burnside引理

    [BZOJ1004][HNOI2008]Cards 题意:把$n$张牌染成$a,b,c$,3种颜色.其中颜色为$a,b,c$的牌的数量分别为$sa,sb,sc$.并且给出$m$个置换,保证这$m$个置 ...

  8. bzoj1004 [HNOI2008]Cards Burnside 引理+背包

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=1004 题解 直接 Burnside 引理就可以了. 要计算不动点的个数,那么对于一个长度为 \ ...

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

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

随机推荐

  1. net.sf.json

    JSONObject package com.itlwc.test;         import net.sf.json.JSONArray;    import net.sf.json.JSONO ...

  2. 29.python环境搭建

    新的笔记本上搭建的python运行环境:1.首先现在下载python版本,我这里拿的是之前用的3.5老版本 2.双击安装(勾选 add Python3.5 to path) 3.检测是否安装成功 4. ...

  3. 【学习】数据规整化:清理、转换、合并、重塑(续)【pandas】

    @合并重叠数据 还有一种数据组合问题不能用简单的合并或连接运算来处理.比如说,你可能有索引全部或部分重叠的两个数据集 使用numpy的where函数,它用于表达一种矢量化的if - else a = ...

  4. 目标检测——Faster R_CNN使用smooth L1作为bbox的回归损失函数原因

    前情提要—— 网上关于目标检测框架——faster r_cnn有太多太好的博文,这是我在组会讲述faster r_cnn这一框架时被人问到的一个点,当时没答上来,于是会下好好百度和搜索一下研究了一下这 ...

  5. CI/CD

    CI/CD 啥是CI/CD CI: continuous integration, 持续集成.就是频繁地把开发的工作提交到主线代码.主要是为了解决集成问题.什么是集成问题呢,白话说,就是从你本地的代码 ...

  6. 关于python的一些想法

    我来自信息管理与信息系统专业,大一学过c语言但不太精通.学习python是为了学会这门新语言,据了解python会慢慢成为主流编程语言. 因为对绘图方面很感兴趣,希望老师能够在课上多讲一些这方面的东西 ...

  7. c#switch语句的用法

    switch条件语句是一种很常用的选择语句,它与if条件语句不同,它只针对某个表达式的值作出判断,从而决定执行哪一段代码. switch条件语句用到的关键字: switch case break de ...

  8. Grafana短信报警实现

    1.阿里云短信服务API2.Jenkins-plugin:Generic Webhook Trigger Plugin 阿里云 下载阿里云短信服务SDK 创建短信服务ACCESS_KEY_ID Gra ...

  9. MySQL8.0.x免安装配置

    目录 概述 下载 配置环境变量 编辑配置文件 初始化MySQL 安装MySQL系统(Windows)服务 初始化MySQL 启动MySQL 修改默认密码 开启远程登录 概述 MySQL从5.7一下子跳 ...

  10. 项目管理第一篇(PROJECT MANAGEMENT A Systems Approach to Planning, Scheduling, and Controlling)

    请把梦想和野心带上,这是我哥对我说的. 几年下来,人浑浑噩噩,梦想和野心像锋利的石头在水中慢慢被磨平,今天就再次记住,不要让焦虑和失望伴随着人生和家庭. 这是H A R O L D K E R Z N ...