bzoj1488 [HNOI2009]图的同构 Burnside 引理
题目传送门
bzoj1815 - [Shoi2006]color 有色图(双倍经验)
题解
暴力
由于在做题之前已经被告知是 Burnside 引理,貌似思考的时候少了一些乐趣啊。
考虑一个置换 \(p\),想要求出这个置换下的不动点的个数。对于一个不动点,若存在一条边 \((a, b)\),一定存在一条边 \((p_a, p_b)\)。
那么考虑一个长度为 \(l\) 的循环,若 \((i, j)\) 是一条 \(i, j\) 均在循环中的点内部的边,那么所有距离等于 \(i, j\) 的距离的点对之间都会有一条边。这样,我们一共可以将这些边分成 \(\frac x2\) 中内部绑定的组。因此,一个长度为 \(l\) 的循环可以产生 \(2^{\frac x2}\) 种方案的贡献。
下面考虑一条两个端点分别在长度为 \(x\) 的循环和 \(y\) 的循环上的边。这样的话,一条边的两个端点在两个循环上走,一直到下一次两个端点都与原来的两个端点重合。显然对于一组点,会走过 \(\operatorname{lcm}(x, y)\) 个点。于是,一共有 \(\frac {xy}{\operatorname{lcm}(x, y)} = \gcd(x, y)\) 组。贡献为 \(2^{\gcd(x, y)}\) 种方案。
优化
但是暴力算法显然会 TLE。
我们一个置换的贡献仅取决于它的每一个循环的长度,因此我们只需要知道它的每一个循环的长度就可以求出贡献了。
于是我们可以使用 Burnside 引理相关题目的常见套路。
我们用 \(B(n)\) 来表示整数划分的方案数。整数划分是指将 \(n\) 划分为若干个无序的正整数的和。当 \(n=100\) 时,\(B(n)\) 大概是 1e6。当 \(n = 40\) 时大概是 \(4e4\) 左右,\(n=45\) 时候大约 \(9e4\)。
因此,由于本题 \(n=45\),如果暴力枚举划分的话,不会超过 \(9e4\) 次。
那么我们如果枚举出 \(n=\sum \limits_{i=1}^k L_i\),那么方案相当于是从 \(n\) 个位置中选出 \(L_1\) 个位置作为第一个循环的成分,再从剩下的 \(n-L_1\) 个位置中选出 \(L_2\) 个……
因此对于这样的划分,其方案数是:
&\binom n{L_1} \binom{n-L_1}{L2} \binom{n-L_1-L_2}{L_2}\cdots\\
=&\frac{n!}{\prod \limits_{i=1}^k Li!}
\end{align*}
\]
(实际上也可以从可重复元素的全排列的角度考虑
然后求出来划分的方案数以后,我们还需要把每一个数放进这些被安排好的位置,使得每一段真的是一个循环。一个循环的第一个数可以考虑连接到这一段剩余的 \(L-1\) 个位置的任何一个位置。然后对于刚刚选择的那个位置,它又可以连接到剩余的 \(L-2\) 个位置。因此对于一个长度为 \(L\) 的循环,它的合法安排数字的方案数为 \((L-1)!\)。
但是还有一个小问题,如果有两个长度相同的置换,他们是没有什么先后区别的,但是在用之前的组合计算方法的时候,会产生先后的区别。这是不应该产生的。因此令 \(C_i\) 表示等于 \(i\) 的 \(L\) 值的个数。我们还需要乘上 \(C_i!\)。
因此对于一个整数划分的总方案数为:
\]
然后就是和暴力一样的计算不动点的个数了。
最终答案为:
\]
下面是代码。由于 \(\gcd\) 和 \(2\) 的正整数幂均可以预处理,因此总的时间复杂度为 \(O(B_n\cdot n^2)\).
#include<bits/stdc++.h>
#define fec(i, x, y) (int i = head[x], y = g[i].to; i; i = g[i].ne, y = g[i].to)
#define dbg(...) fprintf(stderr, __VA_ARGS__)
#define File(x) freopen(#x".in", "r", stdin), freopen(#x".out", "w", stdout)
#define fi first
#define se second
#define pb push_back
template<typename A, typename B> inline char smax(A &a, const B &b) {return a < b ? a = b , 1 : 0;}
template<typename A, typename B> inline char smin(A &a, const B &b) {return b < a ? a = b , 1 : 0;}
typedef long long ll; typedef unsigned long long ull; typedef std::pair<int, int> pii;
template<typename I>
inline void read(I &x) {
int f = 0, c;
while (!isdigit(c = getchar())) c == '-' ? f = 1 : 0;
x = c & 15;
while (isdigit(c = getchar())) x = (x << 1) + (x << 3) + (c & 15);
f ? x = -x : 0;
}
const int N = 200 + 7;
const int P = 997;
int n, ans, hkk;
int inv[N], fac[N], ifac[N << 1], pw[N], gcd[N][N];
int l[N];
inline void sadd(int &x, int y) { x += y; x >= P ? x -= P : x; }
inline int smod(int x) { return x >= P ? x - P : x; }
inline void ycl() {
fac[0] = 1; for (int i = 1; i <= n; ++i) fac[i] = (ll)fac[i - 1] * i % P;
inv[1] = 1; for (int i = 2; i <= n; ++i) inv[i] = (ll)(P - P / i) * inv[P % i] % P;
ifac[0] = 1; for (int i = 1; i <= n; ++i) ifac[i] = (ll)ifac[i - 1] * inv[i] % P;
pw[0] = 1; for (int i = 1; i <= n; ++i) pw[i] = smod(pw[i - 1] << 1);
for (int i = 0; i <= n; ++i) {
for (int j = 0; j <= i; ++j) {
if (!i || !j) gcd[i][j] = i ^ j;
else gcd[i][j] = gcd[j][i % j];
gcd[j][i] = gcd[i][j];
// dbg("%d %d %d\n", i, j, gcd[i][j]);
}
}
}
inline void calc(int div) {
int cnt = 1;
for (int i = 1; i <= l[0]; ++i) cnt = (ll)cnt * pw[l[i] / 2] % P;//, dbg("l[%d] = %d, %d\n", i, l[i], pw[l[i] / 2]);
for (int i = 1; i <= l[0]; ++i)
for (int j = 1; j < i; ++j) cnt = (ll)cnt * pw[gcd[l[i]][l[j]]] % P;//, dbg("*****************");
sadd(ans, (ll)cnt * div % P);
}
inline void dfs(int n, int lim, int div) {
// dbg("n = %d, lim = %d, div = %d\n", n, lim, div);
if (!n) return calc(div);
for (int i = std::min(lim, n); i; --i) {
int df = div;
for (int j = 1; i * j <= n; ++j) {
df = (ll)df * inv[i] % P, l[++l[0]] = i;
dfs(n - i * j, i - 1, (ll)df * ifac[j] % P);
}
while (l[l[0]] == i) --l[0];
}
}
inline void work() {
ycl();
ans = 0;
dfs(n, n, 1);
printf("%d\n", ans);
}
inline void init() {
read(n);
}
int main() {
#ifdef hzhkk
freopen("hkk.in", "r", stdin);
#endif
init();
work();
fclose(stdin), fclose(stdout);
return 0;
}
附录 - bzoj1815 - [Shoi2006]color 有色图 代码
发现只需要把上面的有/无的状态改为 \(k\) 中颜色就可以了。
#include<bits/stdc++.h>
#define fec(i, x, y) (int i = head[x], y = g[i].to; i; i = g[i].ne, y = g[i].to)
#define dbg(...) fprintf(stderr, __VA_ARGS__)
#define File(x) freopen(#x".in", "r", stdin), freopen(#x".out", "w", stdout)
#define fi first
#define se second
#define pb push_back
template<typename A, typename B> inline char smax(A &a, const B &b) {return a < b ? a = b , 1 : 0;}
template<typename A, typename B> inline char smin(A &a, const B &b) {return b < a ? a = b , 1 : 0;}
typedef long long ll; typedef unsigned long long ull; typedef std::pair<int, int> pii;
template<typename I>
inline void read(I &x) {
int f = 0, c;
while (!isdigit(c = getchar())) c == '-' ? f = 1 : 0;
x = c & 15;
while (isdigit(c = getchar())) x = (x << 1) + (x << 3) + (c & 15);
f ? x = -x : 0;
}
const int N = 53 + 7;
int n, k, P, ans;
int inv[N], fac[N], ifac[N << 1], pw[N], gcd[N][N];
int l[N];
inline void sadd(int &x, int y) { x += y; x >= P ? x -= P : x; }
inline int smod(int x) { return x >= P ? x - P : x; }
inline void ycl() {
fac[0] = 1; for (int i = 1; i <= n; ++i) fac[i] = (ll)fac[i - 1] * i % P;
inv[1] = 1; for (int i = 2; i <= n; ++i) inv[i] = (ll)(P - P / i) * inv[P % i] % P;
ifac[0] = 1; for (int i = 1; i <= n; ++i) ifac[i] = (ll)ifac[i - 1] * inv[i] % P;
pw[0] = 1; for (int i = 1; i <= n; ++i) pw[i] = (ll)pw[i - 1] * k % P;
for (int i = 0; i <= n; ++i) {
for (int j = 0; j <= i; ++j) {
if (!i || !j) gcd[i][j] = i ^ j;
else gcd[i][j] = gcd[j][i % j];
gcd[j][i] = gcd[i][j];
// dbg("%d %d %d\n", i, j, gcd[i][j]);
}
}
}
inline void calc(int div) {
int cnt = 1;
for (int i = 1; i <= l[0]; ++i) cnt = (ll)cnt * pw[l[i] / 2] % P;//, dbg("l[%d] = %d, %d\n", i, l[i], pw[l[i] / 2]);
for (int i = 1; i <= l[0]; ++i)
for (int j = 1; j < i; ++j) cnt = (ll)cnt * pw[gcd[l[i]][l[j]]] % P;//, dbg("*****************");
sadd(ans, (ll)cnt * div % P);
}
inline void dfs(int n, int lim, int div) {
// dbg("n = %d, lim = %d, div = %d\n", n, lim, div);
if (!n) return calc(div);
for (int i = std::min(lim, n); i; --i) {
int df = div;
for (int j = 1; i * j <= n; ++j) {
df = (ll)df * inv[i] % P, l[++l[0]] = i;
dfs(n - i * j, i - 1, (ll)df * ifac[j] % P);
}
while (l[l[0]] == i) --l[0];
}
}
inline void work() {
ycl();
ans = 0;
dfs(n, n, 1);
printf("%d\n", ans);
}
inline void init() {
read(n), read(k), read(P);
}
int main() {
#ifdef hzhkk
freopen("hkk.in", "r", stdin);
#endif
init();
work();
fclose(stdin), fclose(stdout);
return 0;
}
bzoj1488 [HNOI2009]图的同构 Burnside 引理的更多相关文章
- BZOJ 1488 Luogu P4727 [HNOI2009]图的同构 (Burnside引理、组合计数)
题目链接 (Luogu) https://www.luogu.org/problem/P4727 (BZOJ) https://www.lydsy.com/JudgeOnline/problem.ph ...
- bzoj1488[HNOI2009]图的同构
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1488 1488: [HNOI2009]图的同构 Time Limit: 10 Sec M ...
- [bzoj1488][HNOI2009]图的同构——Polya定理
题目大意 求两两互不同构的含n个点的简单图有多少种. 简单图是关联一对顶点的无向边不多于一条的不含自环的图. a图与b图被认为是同构的是指a图的顶点经过一定的重新标号以后,a图的顶点集和边集能完全与b ...
- 【BZOJ1488】[HNOI2009]图的同构(Burside引理,Polya定理)
[BZOJ1488][HNOI2009]图的同构(Burside引理,Polya定理) 题面 BZOJ 洛谷 题解 求本质不同的方案数,很明显就是群论这套理论了. 置换一共有\(n!\)个,考虑如何对 ...
- 【BZOJ1488】[HNOI2009]图的同构计数
题目链接 题意 求 n 个点的同构意义下不同的图的数量.\((n\leq 60)\) Sol \(Polya\) 定理的练手题. 我们这里先把边的存在与否变成对边进行黑白染色,白色代表不存在,这样就变 ...
- [bzoj 1004][HNOI 2008]Cards(Burnside引理+DP)
题目:http://www.lydsy.com/JudgeOnline/problem.php?id=1004 分析: 1.确定方向:肯定是组合数学问题,不是Polya就是Burnside,然后题目上 ...
- POJ 2888 Magic Bracelet(Burnside引理,矩阵优化)
Magic Bracelet Time Limit: 2000MS Memory Limit: 131072K Total Submissions: 3731 Accepted: 1227 D ...
- BZOJ 1004 Cards(Burnside引理+DP)
题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1004 题意:三种颜色的扑克牌各有Sr,Sb,Sg张.给出m种置换.两种染色方案在某种置换 ...
- BZOJ_[HNOI2008]_Cards_(置换+Burnside引理+乘法逆元+费马小定理+快速幂)
描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1004 共n个卡片,染成r,b,g三种颜色,每种颜色的个数有规定.给出一些置换,可以由置换得到的 ...
随机推荐
- 【CF1257D】Yet Another Monster Killing Problem【贪心】
题意:给定一些怪物,每天可以选一个勇士进去打怪,每个勇士每天只能打不超过si个怪物,每个勇士只能打能力值≤pi的怪物,问最少多少天打完所有怪物 题解:贪心,每天尽可能多的去打怪,那么存一个对于长度为i ...
- [CSP-S模拟测试]:走格子(模拟+BFS+Dijkstra)
题目描述 $CYJ$想找到他的小伙伴$FPJ$,$CYJ$和$FPJ$现在位于一个房间里,这个房间的布置可以看成一个$N$行$M$列的矩阵,矩阵内的每一个元素会是下列情况中的一种:$1.$障碍区域—这 ...
- Cookie 记录最后访问时间
package cn.gs.ly.servlet; import java.io.IOException; import java.io.PrintWriter; import java.util.D ...
- 【SD系列】SAP 跨年时更改销售凭证号码段
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[SD系列]SAP 跨年时更改销售凭证号码段 ...
- python 按二维数组的某行或列排序 (numpy lexsort)
lexsort支持对数组按指定行或列的顺序排序:是间接排序,lexsort不修改原数组,返回索引. (对应lexsort 一维数组的是argsort a.argsort()这么使用就可以:argsor ...
- (转载)Manacher'sAlgorithm: O(n)时间求字符串的最长回文子串
以下内容转载自:传送门 源于这两篇文章: http://blog.csdn.net/ggggiqnypgjg/article/details/6645824http://zhuhongcheng.wo ...
- LinkedList -链表集合
package cn.learn.collection; import java.util.LinkedList; import java.util.Queue; /* java.util.xxx A ...
- Action 分离
原文地址:http://www.cnblogs.com/giggle/p/5380832.html?utm_source=tuicool&utm_medium=referral 本处摘要备注. ...
- jvm(1)性能监控-linux相关命令
top命令能够实时显示系统中各个进程的资源占用情况,其输出信息分为两部分,前半部分为系统统计信息,后半部分是进程信息. 第一行是任务队列信息,它的结果等同于uptime命令. 第二行是进程统计信息: ...
- java_第一年_JavaWeb(9)
JavaBean是一个遵循某种特定写法的Java类,有以下特点: 必需具有一个无参的构造函数 属性必需私有化 私有化的属性必需通过public类型的方法暴露给其它程序,其方法命名也有一定的规范 范例: ...