UVA11540 Sultan's Chandelier Burnside 引理 + DP
题目传送门
https://vjudge.net/problem/UVA-11540
题解
首先如果两棵树经过一定的旋转以后能够重合,我们就称它们同构。
判断树的同构的常见方法是哈希。但是由于子树的先后顺序是有影响的,只有旋转以后相同才算等价,所以我的哈希方法是直接把所有孩子组成的序列倍长以后,用类似于字符串哈希的方法求出以每个孩子开始一圈的哈希值,取最小的作为当前这棵树的哈希。这样我们就可以轻易的判断两棵子树等不等价了。
然后,我们令 \(dp[x]\) 表示 \(x\) 的子树内,本质不同的染色方案的数量。
根据 Burnside 引理,在求 \(x\) 的方案的时候,我们可以枚举它的孩子们的环的旋转的幅度,然后判断这样旋转合不合法,即——旋转以后对应的两棵子树必须同构。这样,对于一个合法的幅度,这个环一定被分成了一些组,每一组的每棵子树必须使用一样的染色方案。我们把这样的每一组的方案乘起来以后的结果取平均就可以了。
另外,上面是 \(O(n^2)\) 的,实际上可以用 kmp 的最小循环节来线性的求,不详细展开,可以自己研究。
下面是代码,如上文所述,dp 部分的时间复杂度为 \(O(n^2)\),建树为 \(O(n^2)\),因此总的时间复杂度为 \(O(Tn^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 = 100 + 7;
const int P = 1e9 + 7;
const int base = 1997;
int n, k, nod;
int siz[N], dp[N], vis[N];
std::vector<int> g[N];
char s[N * 10];
ull h[N], tmp[N << 1], bin[N];
inline int smod(int x) { return x >= P ? x - P : x; }
inline void sadd(int &x, int y) { x += y; x >= P ? x -= P : x; }
inline int fpow(int x, int y) {
int ans = 1;
for (; y; y >>= 1, x = (ll)x * x % P) if (y & 1) ans = (ll)ans * x % P;
return ans;
}
inline int build(int l, int r) {
// dbg("l = %d, r = %d\n", l, r);
if (r < l) return ++nod;
int o = ++nod, pre = l, cnt = 1;
for (int i = l + 1; i <= r; ++i) {
if (s[i] == '[') {
if (!cnt) pre = i;
++cnt;
} else if (s[i] == ']') {
--cnt;
if (!cnt) g[o].pb(build(pre + 1, i - 1));
}
}
return o;
}
inline void dfs1(int x) {
siz[x] = 1;
int cnt = 0;
for (int y : g[x]) {
dfs1(y);
siz[x] += siz[y], ++cnt;
// tmp[tmp[0] + 1] = tmp[tmp[0]] * base + h[y], ++tmp[0];
}
int tt = 0;
for (int y : g[x]) tmp[tt + 1] = tmp[tt] * base + h[y], ++tt;
for (int y : g[x]) tmp[tt + 1] = tmp[tt] * base + h[y], ++tt;
// dbg("****************** %llu %llu %llu %llu %llu\n", tmp[0], tmp[1], tmp[2], tmp[3], tmp[4]);
for (int i = cnt; i; --i) tmp[i] = tmp[i + cnt] - bin[cnt] * tmp[i];
std::sort(tmp + 1, tmp + cnt + 1);
// dbg("****************** %llu %llu %llu %llu %llu, %llu\n", tmp[0], tmp[1], tmp[2], tmp[3], tmp[4], bin[cnt]);
if (cnt >= 1) h[x] = tmp[1] * siz[x];
else h[x] = 1;
// dbg("******************\n");
// dbg("h[%d] = %llu, siz = %d\n", x, h[x], siz[x]);
}
inline void dfs2(int x) {
int cnt = 0, ms = 0;
dp[x] = 0;
for (int y : g[x]) dfs2(y), ++cnt;
for (int stp = 0; stp < cnt; ++stp) {
int flag = 1, s = 1;
for (int i = 0; i < cnt && flag; ++i) if (!vis[i]) {
for (int j = i; !vis[j] && flag; j += stp, j >= cnt ? j -= cnt : 0) {
vis[j] = 1;
if (h[g[x][i]] != h[g[x][j]]) flag = 0;
}
}
for (int i = 0; i < cnt; ++i) vis[i] = 0;
if (!flag) continue;
for (int i = 0; i < cnt; ++i) if (!vis[i]) {
for (int j = i; !vis[j]; j += stp, j >= cnt ? j -= cnt : 0) vis[j] = 1;
s = (ll)s * dp[g[x][i]] % P;
}
for (int i = 0; i < cnt; ++i) vis[i] = 0;
sadd(dp[x], s), ++ms;
}
// dbg("dp[%d] = %d, ms = %d\n", x, dp[x], ms);
dp[x] = (ll)dp[x] * fpow(ms, P - 2) % P * k % P;
if (!cnt) dp[x] = k;
// dbg("dp[%d] = %d, ms = %d\n", x, dp[x], ms);
}
inline void vec_cls(std::vector<int> &x) {
std::vector<int> y;
std::swap(x, y);
}
inline void ycl() {
bin[0] = 1;
for (int i = 1; i <= n; ++i) bin[i] = bin[i - 1] * base;
}
inline void work() {
nod = -1;
for (int i = 1; i <= n; ++i) vec_cls(g[i]);
build(1, strlen(s + 1));
n = nod;
ycl();
int ans = 1;
for (int i = 1; i <= n; ++i) if (!siz[i]) {
dfs1(i);
dfs2(i);
ans = (ll)ans * dp[i] % P;
}
static int CS = 0;
printf("Case #%d: %d\n", ++CS, ans);
}
inline void init() {
memset(siz, 0, sizeof(siz));
scanf("%s", s + 1);
read(k);
}
int main() {
#ifdef hzhkk
freopen("hkk.in", "r", stdin);
#endif
int T;
read(T);
while (T--) {
init();
work();
}
fclose(stdin), fclose(stdout);
return 0;
}
UVA11540 Sultan's Chandelier Burnside 引理 + DP的更多相关文章
- BZOJ 1004 Cards(Burnside引理+DP)
题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1004 题意:三种颜色的扑克牌各有Sr,Sb,Sg张.给出m种置换.两种染色方案在某种置换 ...
- [BZOJ 1004] [HNOI2008] Cards 【Burnside引理 + DP】
题目链接:BZOJ - 1004 题目分析 首先,几个定义和定理引理: 群:G是一个集合,*是定义在这个集合上的一个运算. 如果满足以下性质,那么(G, *)是一个群. 1)封闭性,对于任意 a, b ...
- BZOJ 1004 Cards(Burnside引理+DP)
因为有着色数的限制,故使用Burnside引理. 添加一个元置换(1,2,,,n)形成m+1种置换,对于每个置换求出循环节的个数, 每个循环节的长度. 则ans=sigma(f(i))/(m+1) % ...
- bzoj1004: [HNOI2008]Cards(burnside引理+DP)
题目大意:3种颜色,每种染si个,有m个置换,求所有本质不同的染色方案数. 置换群的burnside引理,还有个Pólya过几天再看看... burnside引理:有m个置换k种颜色,所有本质不同的染 ...
- bzoj 1004 burnside 引理+DP
对于burnside引理需要枚举染色,这道题属于burnside的一种简单求解的方法,就是polya,我们可以使每一种置换中的循环节中的元素的颜色都相同,那么这样的话就可以直接DP了,我们可以将m个置 ...
- [BZOJ1547]周末晚会:Burnside引理+DP
分析 Attention!这道题的模数是\(1e8+7\). 注意到循环同构会被认为是同一种方案,我们可以把顺时针旋转每个人的位置作为置换,容易发现这些置换一定会形成一个置换群,于是题目所求的所有合法 ...
- [bzoj 1004][HNOI 2008]Cards(Burnside引理+DP)
题目:http://www.lydsy.com/JudgeOnline/problem.php?id=1004 分析: 1.确定方向:肯定是组合数学问题,不是Polya就是Burnside,然后题目上 ...
- BZOJ 1004: [HNOI2008]Cards( 置换群 + burnside引理 + 背包dp + 乘法逆元 )
题意保证了是一个置换群. 根据burnside引理, 答案为Σc(f) / (M+1). c(f)表示置换f的不动点数, 而题目限制了颜色的数量, 所以还得满足题目, 用背包dp来计算.dp(x,i, ...
- 【BZOJ1004】【HNOI2008】Cards 群论 置换 burnside引理 背包DP
题目描述 有\(n\)张卡牌,要求你给这些卡牌染上RGB三种颜色,\(r\)张红色,\(g\)张绿色,\(b\)张蓝色. 还有\(m\)种洗牌方法,每种洗牌方法是一种置换.保证任意多次洗牌都可用这\( ...
随机推荐
- [CF1166C]A Tale of Two Lands题解
比赛的时候lowerbound用错了 现场WA on test4(好吧我承认我那份代码确实有除了lowerbound以外的问题) 于是只能手动二分 (我好菜啊QAQ 经过一波数学推算,我们发现,设序列 ...
- 【HDOJ6623】Minimal Power of Prime(Powerful Number)
题意:给定大整数n,求其质因数分解的最小质数幂 n<=1e18 思路:常规分解算法肯定不行 考虑答案大于1的情况只有3种:质数的完全平方,质数的完全立方,以及p^2*q^3,p,q>=1三 ...
- SQL中MINUS的用法与UNION的用法
一:MINUS指令 其是运用在两个 SQL 语句上.它先找出第一个 SQL 语句所产生的结果,然后看这些结果有没有在第二个 SQL语句的结果中.如果有的话,那第一个SQL结果数据就被去除,而不会在最后 ...
- 安装及启动Tomcat
安装及启动Tomcat 法一:从命令行启动Tomcat: 配置环境变量 Windos+R输入cmd打开dos窗口转到D:\apache-tomcat-7.0.54\bin目录,并输入startup.b ...
- [CSP-S模拟测试]:Rectangle(模拟+树状数组)
题目描述 平面上有$n$个点,第$i$个点的坐标为$X_i,Y_i$.对于其中的一个非空点集$S$,定义$f(S)$为一个最小矩形,满足:$\bullet$覆盖$S$中所有的点(在边界上也算覆盖):$ ...
- vue 如何通过监听路由变化给父级路由菜单添加active样式
1.项目需求:在项目开发中,多级菜单的情况下,勾选子菜单时,需要在父级菜单添加active样式. 2.遇到的问题:一级路由菜单的话,点击当前路由会自动在路由标签上添加router-link-exact ...
- 2019/10/13 TZOJ
水题虽不好,但是很爽 渴望未来某天能把剩下的题补了,先做个记录. Hard Disk Drive http://acm.hdu.edu.cn/showproblem.php?pid=4788 单位转化 ...
- Nginx 介绍配置
nginx的功能和优缺点 nginx是一种服务器软件,将程序放在nginx服务器上,将程序发布出去,nginx是一种高性能的Http和反向代理服务器,同时也是一个代理邮件服务器,也可以实现负载均衡. ...
- eclipse或者myeclipse的代码提示功能
第一步:打开eclipse,在菜单栏选择window-->preferences 第二步:在左边tree菜单栏选择General-->keys 第三步:在右边的文本框中输入content, ...
- java.lang.IllegalStateException: Cannot forward after response has been committed
jjava.lang.IllegalStateException: Cannot forward after response has been committed at org.apache.cat ...