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

#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. 杭州优步uber司机第三组奖励政策

    -8月9日更新- 优步杭州第三组: 定义为激活时间在2015/8/3之后(含)的车主(以优步后台数据显示为准) 滴滴快车单单2.5倍,注册地址:http://www.udache.com/如何注册Ub ...

  2. 宁波Uber优步司机奖励政策(1月11日~1月17日)

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

  3. beego orm mysql

    beego框架中的rom支持mysql 项目中使用到mvc模式,总结下使用方式: models中 package models import ( //使用beego orm 必备 "gith ...

  4. Allure--自动化测试报告生成

    之前尝试使用过testNG自带的测试报告.优化过reportNG的测试报告,对这两个报告都不能满意.后经查找资料,发现有个神器: Allure(已经有allure2了,笔者使用的就是allure2), ...

  5. 打包一个Docker镜像,让你的好友加载开启一个容器,并且每隔一秒输出hello,world到指定的文件中

    一.两个脚本代码 Dockerfile FROM bash COPY . /usr/herui/ WORKDIR /usr/herui/ CMD [ "sh", "hel ...

  6. Mongo DB Java操作

    1.首先下载Mongo DB java 驱动 2.操作Mongo 增删改查 package com.sjjy.mongo; import java.util.ArrayList;import java ...

  7. Vuejs 实现简易 todoList 功能 与 组件

    todoList 结合之前 Vuejs 基础与语法 使用 v-model 双向绑定 input 输入内容与数据 data 使用 @click 和 methods 关联事件 使用 v-for 进行数据循 ...

  8. Leetcode - 557. Reverse Words in a String III (C++) stringstream

    1. 题目:https://leetcode.com/problems/reverse-words-in-a-string-iii/discuss/ 反转字符串中的所有单词. 2. 思路: 这题主要是 ...

  9. Linux内核设计笔记10——内核同步

    Linux内核同步笔记 几个基本概念 - 临界区(critical region):访问和操作共享数据的代码段: - 原子操作:操作在执行中不被打断,要么不执行,要么执行完: - 竞争条件: 两个线程 ...

  10. Ext JS 6学习文档-第4章-数据包

    Ext JS 6学习文档-第4章-数据包 数据包 本章探索 Ext JS 中处理数据可用的工具以及服务器和客户端之间的通信.在本章结束时将写一个调用 RESTful 服务的例子.下面是本章的内容: 模 ...