判断两棵无根树是否同构只需要把重心提作根哈希即可。由于只添加了一个叶子,重心的位置几乎不发生偏移,所以直接把两棵树的重心提起来,逐层找哈希值不同且对应的两子树即可。被一个普及组子问题卡一年。

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std;
#define ll long long
#define ul unsigned long long
#define N 100010
char getc(){char c=getchar();while ((c<'A'||c>'Z')&&(c<'a'||c>'z')&&(c<''||c>'')) c=getchar();return c;}
int gcd(int n,int m){return m==?n:gcd(m,n%m);}
int read()
{
int x=,f=;char c=getchar();
while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
return x*f;
}
int n,ans,f[N],g[N];
struct tree
{
int p[N],t,size[N],root;
ul hax[N],f[N];
struct data{int to,nxt;}edge[N<<];
void addedge(int x,int y){t++;edge[t].to=y,edge[t].nxt=p[x],p[x]=t;}
void dfs(int k,int from)
{
size[k]=;
for (int i=p[k];i;i=edge[i].nxt)
if (edge[i].to!=from)
dfs(edge[i].to,k),size[k]+=size[edge[i].to];
}
int findroot(int k,int s)
{
int mx=;
for (int i=p[k];i;i=edge[i].nxt)
if (size[edge[i].to]<size[k]&&size[edge[i].to]>size[mx]) mx=edge[i].to;
if (size[mx]*>s) return findroot(mx,s);
else return k;
}
void refind()
{
for (int i=p[root];i;i=edge[i].nxt)
if (size[edge[i].to]*>=size[root]) {root=edge[i].to;break;}
}
void gethash(int k,int from)
{
for (int i=p[k];i;i=edge[i].nxt)
if (edge[i].to!=from) gethash(edge[i].to,k);
int cnt=;
for (int i=p[k];i;i=edge[i].nxt)
if (edge[i].to!=from) f[++cnt]=hax[edge[i].to];
sort(f+,f+cnt+);
hax[k]=;
for (int i=;i<=cnt;i++) hax[k]=hax[k]*+f[i];
hax[k]+=size[k]*;
}
}a,b;
bool cmp(const int&x,const int&y)
{
return a.hax[x]<a.hax[y];
}
bool cmp2(const int&x,const int&y)
{
return b.hax[x]<b.hax[y]||b.hax[x]==b.hax[y]&&x>y;
}
void work(int x,int y)
{
int cnt=,cnt2=;
for (int i=a.p[x];i;i=a.edge[i].nxt)
if (a.size[a.edge[i].to]<a.size[x]) f[++cnt]=a.edge[i].to;
for (int i=b.p[y];i;i=b.edge[i].nxt)
if (b.size[b.edge[i].to]<b.size[y]) g[++cnt2]=b.edge[i].to;
sort(f+,f+cnt+,cmp);
sort(g+,g+cnt2+,cmp2);
if (cnt>cnt2||cnt2>cnt+) return;
if (cnt2==cnt+)
{
int t=;
for (int i=,j=;i<=cnt2;i++,j++)
if (a.hax[f[j]]!=b.hax[g[i]]||j>cnt) if (!t) t=g[i++];else return;
if (t) ans=min(ans,t);
return;
}
if (cnt2==cnt)
{
int u,v;
for (u=;u<=cnt;u++) if (a.hax[f[u]]!=b.hax[g[u]]) break;
if (!u) return;
if (a.hax[f[u]]<b.hax[g[u]]) for (v=u;v<cnt;v++) if (a.hax[f[v+]]!=b.hax[g[v]]) break;
if (!v) return;
for (int i=v+;i<=cnt;i++) if (a.hax[f[i]]!=b.hax[g[i]]) return;
for (int i=;i<=cnt;i++) if (b.hax[g[v]]==b.hax[g[i]]) work(f[u],g[i]);
}
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("bzoj4754.in","r",stdin);
freopen("bzoj4754.out","w",stdout);
const char LL[]="%I64d\n";
#else
const char LL[]="%lld\n";
#endif
n=read();ans=n+;
for (int i=;i<n;i++)
{
int x=read(),y=read();
a.addedge(x,y),a.addedge(y,x);
}
for (int i=;i<=n;i++)
{
int x=read(),y=read();
b.addedge(x,y),b.addedge(y,x);
}
a.dfs(,),b.dfs(,);
a.root=a.findroot(,n),b.root=b.findroot(,n+);
a.dfs(a.root,a.root),b.dfs(b.root,b.root);
a.gethash(a.root,a.root),b.gethash(b.root,b.root);
work(a.root,b.root);
if (n&) a.refind();
else b.refind();
a.dfs(a.root,a.root),b.dfs(b.root,b.root);
a.gethash(a.root,a.root),b.gethash(b.root,b.root);
work(a.root,b.root);
cout<<ans;
return ;
}

BZOJ4754 JSOI2016独特的树叶(哈希)的更多相关文章

  1. bzoj4754[JSOI2016]独特的树叶

    这个题....别人写得怎么都....那么短啊? 我怎么....WA了好几次啊....怎么去loj扒了数据才调出来啊? 这个算法...怎么我还是不知道对不对啊 怎么回事啊怎么回事啊怎么回事啊? 请无视上 ...

  2. 【BZOJ4754】独特的树叶(哈希)

    [BZOJ4754]独特的树叶(哈希) 题面 BZOJ 给定一个\(n\)个节点的树A和一个\(n+1\)个节点的树\(B\) 求\(B\)的一个编号最小的节点,使得删去这个节点后\(A,B\)同构 ...

  3. BZOJ 4754 [JSOI2016]独特的树叶 | 树哈希判同构

    题目链接 这道题是一道判断无根树同构的模板题,判断同构主要的思路就是哈希. 一遇到哈希题,一百个人能有一百零一种哈希方式,这篇题解随便选用了一种--类似杨弋<Hash在信息学竞赛中的一类应用&g ...

  4. BZOJ4754 & 洛谷4323 & LOJ2072:[JSOI2016]独特的树叶——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4754 https://www.luogu.org/problemnew/show/P4323 ht ...

  5. P4323 [JSOI2016]独特的树叶(树哈希)

    传送门 树哈希?->这里 反正大概就是乱搞--的吧-- //minamoto #include<bits/stdc++.h> #define R register #define l ...

  6. Luogu 4323 [JSOI2016]独特的树叶

    新技能get 树哈希,考虑到两棵树相同的条件,把每一个结点的哈希值和树的siz写进哈希值里去. 做出A树每一个结点为根时的树的哈希值丢进set中,然后暴力枚举B树中度数为1的点,求出删掉这个点之后的哈 ...

  7. [JSOI2016]独特的树叶

    https://zybuluo.com/ysner/note/1177340 题面 有一颗大小为\(n\)的树\(A\),现加上一个节点并打乱编号,形成树\(B\),询问加上的节点最后编号是多少? \ ...

  8. Luogu P4323 [JSOI2016]独特的树叶

    一道比较好的树Hash的题目,提供一种不一样的Hash方法. 首先无根树的同构判断一般的做法只有树Hash,所以不会的同学可以做了Luogu P5043 [模板]树同构([BJOI2015]树的同构) ...

  9. bzoj 4754: [Jsoi2016]独特的树叶

    不得不说这是神题. %%%   http://blog.csdn.net/samjia2000/article/details/51762811 #include <cstdio> #in ...

随机推荐

  1. Luogu P1802 5倍经验日_KEY

    题目传送门 ·背包 这可以说是一道背包的变形. 首先需要考虑到的是如何将ta转换为一个正常 的背包. 这些数据有一个让我们都十分不爽的地方就是有两个值. 所以我们就设置一个基准值,将失败的经验值当做基 ...

  2. 【BZOJ1564】【NOI2009】二叉查找树(动态规划)

    [BZOJ1564][NOI2009]二叉查找树(动态规划) 题面 BZOJ 洛谷 题目描述 已知一棵特殊的二叉查找树.根据定义,该二叉查找树中每个结点的数据值都比它左儿子结点的数据值大,而比它右儿子 ...

  3. 成都Uber优步司机奖励政策(2月25日)

    滴快车单单2.5倍,注册地址:http://www.udache.com/ 如何注册Uber司机(全国版最新最详细注册流程)/月入2万/不用抢单:http://www.cnblogs.com/mfry ...

  4. WeTest功能优化第2期:云真机智能投屏,调试告别鼠标

    第2期功能优化目录 [云真机视频映射]云真机画面本地映射[兼容性测试报告]新增问题机型聚类功能[新增Android9.0]同步上线最新安卓系统 本期介绍的云测产品功能优化,既有重磅级技术突破,也有报告 ...

  5. APP上下左右滑动屏幕的处理

    #获得机器屏幕大小x,y driver = self.driver def getSize(): x = driver.get_window_size()['width'] y = driver.ge ...

  6. CF245H Queries for Number of Palindromes

    题目描述 给你一个字符串s由小写字母组成,有q组询问,每组询问给你两个数,l和r,问在字符串区间l到r的字串中,包含多少回文串. 时空限制 5000ms,256MB 输入格式 第1行,给出s,s的长度 ...

  7. TW实习日记:第31-32天

    不知不觉的,实习的净工作天数,已经都超过一个月了.因为对工作内容不是很满意,所以打算月底离职,也不知道是公司太缺人还是我真的能干活,领导竟然三番两次找我让我再考虑...明天又要找我了,哎...随机应变 ...

  8. JS原型链与继承别再被问倒了

    原文:详解JS原型链与继承 摘自JavaScript高级程序设计: 继承是OO语言中的一个最为人津津乐道的概念.许多OO语言都支持两种继承方式: 接口继承 和 实现继承 .接口继承只继承方法签名,而实 ...

  9. python中的迭代器与生成器

    迭代器 迭代器的引入 假如我现在有一个列表l=['a','b','c','d','e'],我想取列表中的内容,那么有几种方式? 1.通过索引取值 ,如了l[0],l[1] 2.通过for循环取值 fo ...

  10. Android开发-API指南-<permission>

    <permission> 英文原文:http://developer.android.com/guide/topics/manifest/permission-element.html 采 ...