首页 技术 正文
技术 2022年11月21日
0 收藏 533 点赞 5,106 浏览 2601 个字

【BZOJ1478】Sgu282 Isomorphism

题意:用$m$种颜色去染一张$n$个点的完全图,如果一个图可以通过节点重新标号变成另外一个图,则称这两个图是相同的。问不同的染色方案数。答案对$P$取模。

$n\le 53,m\le 1000,P>n,P$是质数。

题解:对于本题来说,每个元素是所有边,每个置换是边的置换,而边的置换难以表示,点的置换容易表示,所以我们考虑点置换和边置换的关系。

如果两个点置换有着相同的结构,则它们对应的边置换的循环数相同。

$$\begin{pmatrix}1&2&3 \ 2&1&3\end{pmatrix}\rightarrow\begin{pmatrix}(1,2)&(1,3)&(2,3) \ (1,2)&(2,3)&(1,3)\end{pmatrix}$$

$$\begin{pmatrix}1&2&3 \ 1&3&2\end{pmatrix}\rightarrow\begin{pmatrix}(1,2)&(1,3)&(2,3) \ (1,3)&(1,2)&(2,3)\end{pmatrix}$$

一个点置换的结构:在表示成循环节时,循环的大小依次为$L_1,L_2,…L_K$。

$L_1,L_2,…L_K$可以组成一个$n$的划分。

而当$n$=53时,$n$个划分数不超过$300000$。

可以通过搜索得出。

那么一个结构为$L_1,L_2,…L_K$的点置换对应边置换的循环数是什么呢?

对于端点在不同点循环中的边,设两端点所在点循环大小为$L_1,L_2$,则这样的边一共有$L_1\times L_2$条,而每个循环有$lcm(L_1,L_2)$条边,所以一共有$gcd(L_1,L_2)$个循环。

对于端点在同一点循环中的边,设所在点循环的大小为$L$,这样的边一共有$C_L^2$条。然后分奇偶讨论:

  1. 如果$L$是奇数,那么一个循环中有$L$条边,所以循环数为${C_L^2\over L}={L-1\over 2}$。
  2. 如果$L$是偶数,那么一个循环中有$L$条边,但是如果两端点相隔$L\over 2$,这个循环中有$L\over 2$条边。所以循环数为${C_L^2-{L\over 2}\over L}+1={L\over 2}$。

所以一个大小为$L$的点循环中边循环的数量是$\lfloor{L\over 2}\rfloor$。

结构为$L_1,L_2,…L_K$的点置换中边循环的数量就是

$$\sum\limits_{i=1}^n\lfloor{L_i\over 2}\rfloor+\sum\limits_{i=1}K\sum\limits_{j=1}{i-1}gcd(L_i,L_j)$$

那么有多少结构为$L_1,L_2…L_K$的点置换呢?可以先求出$n!\over {L_1!L_2!…L_K!}$得到每个点属于哪个循环的方案数,而对于每个循环,我们可以先确定第一个点,然后其余点任意排列,方案数是$(L_1-1)!(L_2-1)!…(L_K-1)!$。乘起来得到$n!\over {L_1L_2…L_k}$。

又由于存在一些$L$相等的情况,设有$t$种本质不同的$L$,每种个数为$B_i$,所以总方案数还要除掉$B_1!B_2!…B_t!$,所以总数为:

$$n!\over L_1L_2…L_KB_1!B_2!…B_t!$$

最后套上Pólya定理,令$c=\sum\limits_{i=1}^n\lfloor{L_i\over 2}\rfloor+\sum\limits_{i=1}K\sum\limits_{j=1}{i-1}gcd(L_i,L_j)$,$s={n!\over L_1L_2…L_KB_1!B_2!…B_t!}$,答案就是

$$\frac 1 {n!}\sum s\times m^c$$

#include <cstdio>
#include <cstring>
#include <iostream>
using namespace std;
typedef long long ll;
ll P,ans;
int n,m;
ll jc[70],jcc[70],ine[70],pw[3700];
int g[70][70],v[70];
inline int rd()
{
int ret=0,f=1;char gc=getchar();
while(gc<'0'||gc>'9'){if(gc=='-')f=-f;gc=getchar();}
while(gc>='0'&&gc<='9')ret=ret*10+(gc^'0'),gc=getchar();
return ret*f;
}
inline ll C(int a,int b)
{
if(a<b)return 0;
return jc[a]*jcc[a-b]%P*jcc[b]%P;
}
void dfs(int len,int sum)
{
if(!sum)
{
ll s=jc[n],c=0;
int i,j,tmp;
for(i=1,tmp=0;i<=len;i++)
{
tmp++;
if(i==len||v[i]!=v[i+1])s=s*jcc[tmp]%P,tmp=0;
s=s*ine[v[i]]%P;
c+=v[i]>>1;
for(j=1;j<i;j++)c+=g[v[j]][v[i]];
}
ans=(ans+pw[c]*s)%P;
return ;
}
for(int i=min(sum,v[len]);i>=1;i--)v[len+1]=i,dfs(len+1,sum-i);
}
int main()
{
n=rd(),m=rd(),P=rd();
int i,j;
for(pw[0]=i=1;i<=n*n/2;i++)pw[i]=pw[i-1]*m%P;
for(i=0;i<=n;i++)g[i][0]=i;
for(i=0;i<=n;i++)for(j=1;j<=i;j++)g[i][j]=g[j][i%j];
jc[1]=ine[1]=jcc[1]=jc[0]=ine[0]=jcc[0]=1;
for(i=2;i<=n;i++)jc[i]=jc[i-1]*i%P,ine[i]=P-(P/i)*ine[P%i]%P,jcc[i]=jcc[i-1]*ine[i]%P;
v[0]=n;
dfs(0,n);
printf("%lld",ans*jcc[n]%P);
return 0;
}
相关推荐
python开发_常用的python模块及安装方法
adodb:我们领导推荐的数据库连接组件bsddb3:BerkeleyDB的连接组件Cheetah-1.0:我比较喜欢这个版本的cheeta…
日期:2022-11-24 点赞:878 阅读:8,903
Educational Codeforces Round 11 C. Hard Process 二分
C. Hard Process题目连接:http://www.codeforces.com/contest/660/problem/CDes…
日期:2022-11-24 点赞:807 阅读:5,428
下载Ubuntn 17.04 内核源代码
zengkefu@server1:/usr/src$ uname -aLinux server1 4.10.0-19-generic #21…
日期:2022-11-24 点赞:569 阅读:6,245
可用Active Desktop Calendar V7.86 注册码序列号
可用Active Desktop Calendar V7.86 注册码序列号Name: www.greendown.cn Code: &nb…
日期:2022-11-24 点赞:733 阅读:6,057
Android调用系统相机、自定义相机、处理大图片
Android调用系统相机和自定义相机实例本博文主要是介绍了android上使用相机进行拍照并显示的两种方式,并且由于涉及到要把拍到的照片显…
日期:2022-11-24 点赞:512 阅读:7,688
Struts的使用
一、Struts2的获取  Struts的官方网站为:http://struts.apache.org/  下载完Struts2的jar包,…
日期:2022-11-24 点赞:671 阅读:4,726