P4323-[JSOI2016]独特的树叶【换根dp,树哈希】
正题
题目链接:https://www.luogu.com.cn/problem/P4323
题目大意
给出\(n\)个点的树和加上一个点之后的树(编号打乱)。
求多出来的是哪个点(如果有多少个就输出编号最小的)。
\(1\leq n\leq 10^5\)
解题思路
定义一下\(hash\)值\(P(i)\)
我的做法是\(P(i)=p^i\),\(p\)是一个质数,当然这样好像容易被卡,安全点的做法是用第\(i\)个质数或者直接用复数\(hash\)。
然后定义一下带根的\(hash\)值
\]
然后换根求出两棵树里面所有点作为根的\(hash\)值,第一棵树里的丢进\(map\)。然后第二棵树里面枚举一下叶子然后算一下去掉之后的\(hash\)值在\(map\)里面匹配(1号点要特判)
时间复杂度\(O(n\log n)\)
code
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#include<map>
#define ll long long
using namespace std;
const ll N=1e5+10,P=998244353,p=133331;
vector<ll> T[N],G[N];map<ll,bool>mp;
ll n,siz[N],pw[N],f[N],g[N];
ll power(ll x,ll b){
ll ans=1;
while(b){
if(b&1)ans=ans*x%P;
x=x*x%P;b>>=1;
}
return ans;
}
void Dfs1(ll x,ll fa,vector<ll> *T){
siz[x]=f[x]=1;
for(ll i=0;i<T[x].size();i++){
ll y=T[x][i];
if(y==fa)continue;
Dfs1(y,x,T);siz[x]+=siz[y];
(f[x]+=f[y]*pw[siz[y]]%P)%=P;
}
return;
}
void Dfs2(ll x,ll fa,vector<ll> *T){
for(ll i=0;i<T[x].size();i++){
ll y=T[x][i];
if(y==fa)continue;
g[y]=(g[x]-f[y]*pw[siz[y]]%P+P)%P;
(g[y]=f[y]+g[y]*pw[siz[1]-siz[y]]%P)%=P;
Dfs2(y,x,T);
}
return;
}
signed main()
{
scanf("%lld",&n);pw[0]=1;
for(ll i=1;i<=n+1;i++)
pw[i]=pw[i-1]*p%P;
for(ll i=1;i<n;i++){
ll x,y;
scanf("%lld%lld",&x,&y);
G[x].push_back(y);
G[y].push_back(x);
}
for(ll i=1;i<=n;i++){
ll x,y;
scanf("%lld%lld",&x,&y);
T[x].push_back(y);
T[y].push_back(x);
}
Dfs1(1,1,G);g[1]=f[1];Dfs2(1,1,G);
for(ll i=1;i<=n;i++)mp[g[i]]=1;
Dfs1(1,1,T);g[1]=f[1];
Dfs2(1,1,T);
if(T[1].size()<=1){
ll y=T[1][0];
if(mp[f[y]])
{puts("1");return 0;}
}
for(ll x=2;x<=n+1;x++){
if(T[x].size()>1)continue;
ll y=T[x][0];
ll tmp=(g[y]-pw[1]+P)%P;
if(mp[tmp]){printf("%lld\n",x);return 0;}
}
return 0;
}
P4323-[JSOI2016]独特的树叶【换根dp,树哈希】的更多相关文章
- 换根DP+树的直径【洛谷P3761】 [TJOI2017]城市
P3761 [TJOI2017]城市 题目描述 从加里敦大学城市规划专业毕业的小明来到了一个地区城市规划局工作.这个地区一共有ri座城市,<-1条高速公路,保证了任意两运城市之间都可以通过高速公 ...
- Luogu P4323 [JSOI2016]独特的树叶
一道比较好的树Hash的题目,提供一种不一样的Hash方法. 首先无根树的同构判断一般的做法只有树Hash,所以不会的同学可以做了Luogu P5043 [模板]树同构([BJOI2015]树的同构) ...
- P4323 [JSOI2016]独特的树叶(树哈希)
传送门 树哈希?->这里 反正大概就是乱搞--的吧-- //minamoto #include<bits/stdc++.h> #define R register #define l ...
- [BZOJ4379][POI2015]Modernizacja autostrady[树的直径+换根dp]
题意 给定一棵 \(n\) 个节点的树,可以断掉一条边再连接任意两个点,询问新构成的树的直径的最小和最大值. \(n\leq 5\times 10^5\) . 分析 记断掉一条边之后两棵树的直径为 \ ...
- 2018.10.15 NOIP训练 水流成河(换根dp)
传送门 换根dp入门题. 貌似李煜东的书上讲过? 不记得了. 先推出以1为根时的答案. 然后考虑向儿子转移. 我们记f[p]f[p]f[p]表示原树中以ppp为根的子树的答案. g[p]g[p]g[p ...
- [JSOI2016]独特的树叶
https://zybuluo.com/ysner/note/1177340 题面 有一颗大小为\(n\)的树\(A\),现加上一个节点并打乱编号,形成树\(B\),询问加上的节点最后编号是多少? \ ...
- 小奇的仓库:换根dp
一道很好的换根dp题.考场上现场yy十分愉快 给定树,求每个点的到其它所有点的距离异或上m之后的值,n=100000,m<=16 只能线性复杂度求解,m又小得奇怪.或者带一个log像kx一样打一 ...
- 国家集训队 Crash 的文明世界(第二类斯特林数+换根dp)
题意 题目链接:https://www.luogu.org/problem/P4827 给定一棵 \(n\) 个节点的树和一个常数 \(k\) ,对于树上的每一个节点 \(i\) ,求出 \( ...
- Acesrc and Travel(2019年杭电多校第八场06+HDU6662+换根dp)
题目链接 传送门 题意 两个绝顶聪明的人在树上玩博弈,规则是轮流选择下一个要到达的点,每达到一个点时,先手和后手分别获得\(a_i,b_i\)(到达这个点时两个人都会获得)的权值,已经经过的点无法再次 ...
随机推荐
- AspNetCore WebApi
需求 前几天,马老板给小明和小红一个"待办事项"网站,小明负责后端,小红负责前端,并要求网站可以同时在 Windows.和 Linux 上运行. 小明整理了一下"待办事项 ...
- Mybatis--级联(一)
级联是resultMap中的配置. 级联分为3种 鉴别器(discrimination):根据某些条件采用具体实现具体实现类级联,如体检表根据性别去区分 一对一:学生和学生证 一对多:班主任和学生. ...
- Sparksql 日期加减函数
1.日期加减spark sql 日期加减,date_sub,date_addval dateDF=spark.range(10) .withColumn("today",curre ...
- shiro加密流程
- python matplotlib 绘图+显示数值
参考:https://www.jb51.net/article/152685.htm 用plt.text函数 import numpy as np import matplotlib.mlab as ...
- mysql绕过root密码登录
绕过密码登录步骤: 一.Mysql8.0之前: 关闭服务 执行参数:mysqld --skip-grant-tables 新开窗口执行mysql,即可进入mysql 二.Mysql8.0之前: 关闭服 ...
- rsync 服务搭建
rsync 服务搭建 服务端部署操作内容: 创建rsync用户和用户组 eg: useradd -s /sbin/nologin -M rsync 创建需要备份的指定目录,并修改权限 eg: mkdi ...
- vue七种实现组建通信的方法
目录 组件通信 1.props 父组件--->子组件通信 2.$emit 子组件--->父组件传递 $emit与props结合 兄弟组件传值 3.bus(事件总线) 兄弟组件通信 4.$p ...
- SQL-INSERT触发器练习
&练习一 有这样的一个基础表A,字段包括:id.type.value.create_time,主要是记录某个类型的状态变化时间和值.在插入类型(type)为'runtime' 的数据时,根据前 ...
- bean的作用域和生命周期
一.Bean作用域 二.生命周期 其中,这个类实现各种接口重写各种方法,会按bean的声明周期按序执行: 其中,自定义的初始化和自定义销毁的方法不是实现接口重写,而是成员方法,并且在装配bean即在x ...