Description

给定一个N,N<=50 000个节点的仙人掌,其是指每条边最多在一个环中的无向图,求仙人掌有多少种自同构。自同构是指得是图的顶点集合V到V的变换M,

以P1^a1*P2^a2...Pk^ak的形式输出,其中Pk是素数。

建圆方树,找到重心,如果重心有两个就在它们之间的边上插一个点,同构变换后重心不变

以重心为根得到有根树,对于树点,统计交换子树造成的同构,将所有贡献累乘得到答案,对于环点,如果不是重心,就只有翻转这个环能产生新的同构,否则这个环除了翻转还可以旋转,用hash处理这些情况。注意处理环的时候需要把环点的相邻点排成原来在仙人掌上的顺序。

#include<cstdio>
#include<algorithm>
typedef unsigned long long u64;
const int N=;
int n,m,fac[N];
int es[N],enx[N],e0[N],e1[N],ep=;
bool _c[N];
void ae(int*e,int a,int b){
es[ep]=b;enx[ep]=e[a];e[a]=ep++;
es[ep]=a;enx[ep]=e[b];e[b]=ep++;
}
void de(int*e,int a,int b){
for(int*i=e+a;*i;i=enx+*i){
int u=es[*i];
if(u==b){
*i=enx[*i];
return;
}
}
}
int dfn[N],low[N],tk=,ss[N],sp=,n1;
void tj(int w,int pa){
dfn[w]=low[w]=++tk;
ss[++sp]=w;
for(int i=e0[w];i;i=enx[i]){
int u=es[i];
if(!u)continue;
if(!dfn[u]){
es[i^]=;
tj(u,w);
es[i^]=w;
if(ss[sp]==u)--sp,ae(e1,w,u);
}else if(dfn[u]<dfn[w]){
int rp=;
ae(e1,u,++n1);
_c[n1]=;
while(sp&&dfn[ss[sp]]>dfn[u])ae(e1,ss[sp--],n1);
}
}
}
int sz[N],cg[],cgp=,rt;
void f1(int w,int pa){
bool is=;
sz[w]=;
for(int i=e1[w];i;i=enx[i]){
int u=es[i];
if(u==pa)continue;
f1(u,w);
sz[w]+=sz[u];
if(sz[u]*>n1)is=;
}
if(sz[w]*<n1)is=;
if(is)cg[cgp++]=w;
}
u64 h[N],h1[N],h2[N];
int cs[N],cp;
bool cmp_h(int a,int b){return h[a]<h[b];}
int a1[N],a2[N],mx=;
u64 a3[N];
void mul(int x){
++a1[x];
if(x>mx)mx=x;
}
void mulfac(int x){
++a2[x];
if(x>mx)mx=x;
}
void f2(int w,int pa){
for(int i=e1[w];i;i=enx[i]){
int u=es[i];
if(u!=pa)f2(u,w);
}
cp=;
for(int i=e1[w],d=;i;i=enx[i]){
int u=es[i];
if(u==pa){
d=;
continue;
}
if(d)cs[cp++]=u;
}
for(int i=e1[w];i;i=enx[i]){
int u=es[i];
if(u==pa)break;
cs[cp++]=u;
}
if(_c[w]){
if(w==rt){
u64 p0=,pp=;
for(int i=;i<=cp;++i){
h1[i]=h1[i+cp]=h2[i]=h2[i+cp]=h[cs[i-]];
pp*=p0;
}
for(int i=;i<=cp*;++i)h1[i]+=h1[i-]*p0;
for(int i=cp*;i;--i)h2[i]+=h2[i+]*p0;
int c=;
for(int i=cp;i<cp*;++i)c+=(h1[i]-h1[i-cp]*pp==h1[cp]);
for(int i=cp;i;--i)c+=(h2[i]-h2[i+cp]*pp==h1[cp]);
mul(c);
}else{
u64 h1=,h2=;
for(int i=;i<cp;++i)h1=h1*+h[cs[i]];
for(int i=cp-;i>=;--i)h2=h2*+h[cs[i]];
if(h1==h2)mul();
h[w]=std::min(h1,h2);
h[w]^=h[w]>>^h[w]*<<^41546541735416351llu;
}
}else{
std::sort(cs,cs+cp,cmp_h);
h[w]=;
for(int i=,j=;i<cp;i=j){
for(;j<cp&&h[cs[i]]==h[cs[j]];++j);
mulfac(j-i);
}
for(int i=;i<cp;++i)h[w]=h[w]*+h[cs[i]];
h[w]^=h[w]>>^h[w]*<<^12218653252112541llu;
}
}
int _(){
int x=,c=getchar();
while(c<)c=getchar();
while(c>)x=x*+c-,c=getchar();
return x;
}
int ps[N],pp,mp[N],ed[N];
bool np[N];
int main(){
n=_();m=_();
n1=n;
for(int i=;i<m;++i){
pp=_();
for(int j=;j<pp;++j)ps[j]=_();
for(int j=;j<pp-;++j)ae(e0,ps[j],ps[j+]);
}
for(int w=;w<=n;++w){
for(int i=e0[w];i;i=enx[i]){
int u=es[i];
if(ed[u]!=w)ed[u]=w;
else if(u>w)--a1[];
}
}
tj(,);
f1(,);
if(cgp==){
if(_c[cg[]])rt=cg[];
else if(_c[cg[]])rt=cg[];
else{
rt=++n1;
ae(e1,n1,cg[]);
ae(e1,n1,cg[]);
de(e1,cg[],cg[]);
de(e1,cg[],cg[]);
}
}else rt=cg[];
f1(rt,);
f2(rt,);
for(int i=mx;i;--i)a2[i]+=a2[i+];
for(int i=mx;i;--i)a2[i]+=a1[i];
pp=;
for(int i=;i<=mx;++i){
if(!np[i])ps[pp++]=mp[i]=i;
for(int j=;j<pp&&i*ps[j]<=mx;++j){
np[i*ps[j]]=;
mp[i*ps[j]]=ps[j];
if(i%ps[j]==)break;
}
}
for(int i=;i<=mx;++i){
for(int x=i;x>;x/=mp[x])a3[mp[x]]+=a2[i];
}
pp=;
for(int i=;i<=mx;++i)if(a3[i])ps[pp++]=i;
printf("%d\n",pp);
for(int j=;j<pp;++j)printf("%d %lld\n",ps[j],a3[ps[j]]);
return ;
}

bzoj3871: [Neerc2013 C]Cactus Automorphisms || 3899: 仙人掌树的同构的更多相关文章

  1. BZOJ3899 仙人掌树的同构(圆方树+哈希)

    考虑建出圆方树.显然只有同一个点相连的某些子树同构会产生贡献.以重心为根后(若有两个任取一个即可),就只需要处理子树内部了. 如果子树的根是圆点,其相连的同构子树可以任意交换,方案数乘上同构子树数量的 ...

  2. 03-树1 树的同构 (C语言链表实现)

    #include <stdio.h> #include <stdlib.h> #include <string.h> #include <stdbool.h& ...

  3. PAT 03-树1 树的同构 (25分)

    给定两棵树T1和T2.如果T1可以通过若干次左右孩子互换就变成T2,则我们称两棵树是"同构"的.例如图1给出的两棵树就是同构的,因为我们把其中一棵树的结点A.B.G的左右孩子互换后 ...

  4. BZOJ 4337: BJOI2015 树的同构 树hash

    4337: BJOI2015 树的同构 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=4337 Description 树是一种很常见的数 ...

  5. SDUT 3340 数据结构实验之二叉树一:树的同构

    数据结构实验之二叉树一:树的同构 Time Limit: 1000MS Memory Limit: 65536KB Submit Statistic Problem Description 给定两棵树 ...

  6. PTA 深入虎穴 (正解)和树的同构

    在上一篇博客中分享了尝试用单链表修改程序,虽然在Dev上运行没有错误,但是PTA设置的测试点有几个没有通过,具体不清楚问题出现在哪里,所以现在把之前正确的程序放在这里. 7-2 深入虎穴 (30 分) ...

  7. 4337: BJOI2015 树的同构

    题解: 树的同构的判定 有根树从根开始进行树hash 先把儿子的f进行排序 $f[i]=\sum_{j=1}^{k} { f[j]*prime[j]} +num[i]$(我没有仔细想这样是不是树是唯一 ...

  8. [BJOI2015]树的同构

    嘟嘟嘟 判断树的同构的方法就是树上哈希. 如果树是一棵有根树,那么只要从根节点出发dfs,每一个节点的哈希值等于按传统方式算出来的子树的哈希值的结果.需要注意的是,算完子树的哈希值后要先排序再加起来, ...

  9. bzoj4337树的同构

    树是一种很常见的数据结构. 我们把N个点,N-1条边的连通无向图称为树. 若将某个点作为根,从根开始遍历,则其它的点都有一个前驱,这个树就成为有根树. 对于两个树T1和T2,如果能够把树T1的所有点重 ...

随机推荐

  1. Flask初级(二)为flash创建路由,访问路径。

    Project name :Flask_Plan 上一篇文章,我们创建了默认的flask项目,也可以运行起来. 但是只有一个首页,只显示一个hello world. 现在我们创建访问路由,也就是访问地 ...

  2. scroll事件的优化以及scrollTop的兼容性

    scrollTop的兼容性 scroll事件,当用户滚动带滚动条的元素中的内容时,在该元素上面触发.<body>元素中包含所加载页面的滚动条. 虽然scroll事件是在window对象上发 ...

  3. bga

    本文记录在bga布线的难题. 1一开始就要预留好布线局域. 最近出现布线太密,修改时就麻烦了. http://bbs.elecfans.com/jishu_521995_1_1.html

  4. WebGL编程指南案例解析之平移和旋转的math库实现

    这里说的math库实现,指的是,通过一般的加减乘除(角度计算)来更新坐标值. 因为涉及到坐标的变化,所以这里都是基于对顶点着色器的修改 平移: var vShader = ` attribute ve ...

  5. CSS3之border-image

    先上效果图,类似于IPHONE手机左上角的返回按钮样式,如果是在CSS2那么就要做一张背景图片扩展就没那么灵活了,CSS3内就不需要了,CSS3样式挺强大方便的. 源图片: 样式: .banner { ...

  6. CentOS 下安装和使用 Docker

    引言: 在服务器开发过程中,环境部署无疑是及其繁琐的事情,特别是当项目数量和规模达到一定级别之后,在一台新的机器上部署项目环境无疑是极其漫长而痛苦的,那么什么办法能够实现我们的目标:在开发环境的一次配 ...

  7. HDU 4864

    http://acm.hdu.edu.cn/showproblem.php?pid=4864 #include <iostream> #include <cstdio> #in ...

  8. 故障排查:vsftpd无法用浏览器访问

    在CentOS6上搭建的ftp服务器,突然无法使用浏览器进行访问,但使用xftp等工具可以正常访问 想到之前修改过阿里云的安全组设置,推测可能有关 1)修改vsftpd的配置,手动指定被动模式的随机连 ...

  9. rar ubuntu

    http://jingyan.baidu.com/article/1612d5004095eee20e1eeeab.html sudo 7z x ***.rar

  10. 一次delete基表回收DBA权限的危险性操作

    1.0事件还原:测试库某个对象,使用具有DBA角色用户导出,生产环境普通用户无法导入数据,因此需要回收测试库dba角色,revoke无法回收 2.0操作流程 回收角色报错 测试环境测试,delete删 ...