码迷,mamicode.com
首页 > 其他好文 > 详细

独特的树叶

时间:2020-01-03 21:11:46      阅读:76      评论:0      收藏:0      [点我收藏+]

标签:输入   公式   sdi   整数   main   内存限制   +=   sig   void   

1695:独特的树叶

时间限制: 1000 ms         内存限制: 262144 KB

【题目描述】

JYY有两棵树A和B:树A有N个点,编号为1到N;树B有N+1个节点,编号为1到N+1。

JYY知道树B恰好是由树A加上一个叶节点,然后将节点的编号打乱后得到的。他想知道,这个多余的叶子到底是树B中的哪一个叶节点呢?


【输入】

输入一行包含一个正整数N。

接下来N-1行,描述树A,每行包含两个整数表示树A中的一条边;

接下来N行,描述树B,每行包含两个整数表示树B
中的一条边。
【输出】

输出一行一个整数,表示树B中相比树A多余的那个叶子的编号。如果有多个符合要求的叶子,输出B中编号最小的那一个的编号。

 

【数据规模】

1<=n<=1e5.

【题解】

判断两棵无根树是否同构,则需在它们中分别有一个根u,和根v,使得树A以u为根的哈希值等于B以v为根的哈希值。

那么我们只需要求出A树以每一个点为根的哈希值装入map中,在判断B去掉了一个叶子节点(入出度为1)后以任一根的哈希值是否在map中即可。

考虑有根树哈希值。

这里的公式为:

将u的子节点按f值大小排序,令ch[i]=131的i次方。

f[u]=siz[u]*∑f[v]*ch[i-1];(i:1--u子节点个数)

那么我们可以在O(nlogn)时间内求出以1号点为根的所有f值。

现在要求出以任一点为根的根的f值.

考虑换根树形DP。

设g[u]为以u的父亲为根时,去掉u及其子树中所有点后的哈希值。(注意是以u的父亲为根

则对于u的儿子节点

令hu数组表示除去f[v]外的u所有儿子的f值,如果u不为1,则加入g[u].

g[v]=(n-siz[v])∑hu[i]*ch[i-1]。

排序后对hu数组求一个前缀和和后缀和即可O(1)求出每个g[v]。

ans[u]表示以u为根的整棵树哈希值。

ans[u]即为g[v]式子不去掉f[v]所求出的值。

将每个ans加入map中。

对于B树的每个叶节点,去掉它后树的哈希值则为以它为根节点的ans/(n+1) 取质数模数求逆元可得   (有哈希定义得到)

判断得求出的值在map中就输出。

代码如下:

 

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N=1e5+5;
const int mo=1e9+9;
const int cc=131;
int gsn,huan[N],pre[N],sub[N],ch[N];
map <int,int> ma;
struct pigu
{
    int dao,ne;
};
struct choujiao
{
    int hao,zhi;
}dahuan[N];
inline bool cmp(choujiao x,choujiao y)
{
    return x.zhi<y.zhi;
}
struct tree
{
    int n,size,last[N],siz[N],cnt[N],g[N];
    int f[N],ans[N];
    pigu a[N<<1];
    inline void lingjiebiao(int x,int y)
    {
        a[++size].dao=y;
        a[size].ne=last[x];
        last[x]=size;
        cnt[x]++;
    }
    inline void get_f(int now,int fa)
    {
        siz[now]=1;
        int cnt=0;
        for(int i=last[now];i;i=a[i].ne)
        {
            if(a[i].dao==fa) continue;
            get_f(a[i].dao,now);
            siz[now]+=siz[a[i].dao];
        }
        for(int i=last[now];i;i=a[i].ne)
        {
            if(a[i].dao==fa) continue;
            huan[++cnt]=f[a[i].dao];
        }
        sort(huan+1,huan+cnt+1);
        for(int i=1;i<=cnt;i++)
            f[now]+=(huan[i]*ch[i-1])%mo;
        f[now]*=siz[now];f[now]%=mo;
        if(!cnt) f[now]=1;
    }
    inline void get_ans(int now,int fa)
    {
        int cnt=0;
        for(int i=last[now];i;i=a[i].ne)
        {
            if(a[i].dao==fa) continue;
            dahuan[++cnt]={a[i].dao,f[a[i].dao]};
        }
        if(fa) dahuan[++cnt]={-1,g[now]};
        sort(dahuan+1,dahuan+cnt+1,cmp);
        pre[0]=sub[cnt+1]=0;
        for(int i=1;i<=cnt-1;i++) pre[i]=(pre[i-1]+(dahuan[i].zhi*ch[i-1])%mo)%mo;
        for(int i=cnt;i>=2;i--) sub[i]=(sub[i+1]+(dahuan[i].zhi*ch[i-2])%mo)%mo;
        for(int i=1;i<=cnt;i++)
        {
            if(dahuan[i].hao==-1) continue;
            g[dahuan[i].hao]=((n-siz[dahuan[i].hao])*(pre[i-1]+sub[i+1])%mo+mo)%mo;
            g[dahuan[i].hao]%=mo;
        }
        if(!fa&&cnt==1) g[dahuan[1].hao]=1;
        for(int i=1;i<=cnt;i++)
            ans[now]=(ans[now]+dahuan[i].zhi*ch[i-1]%mo)%mo;
        ans[now]*=n;ans[now]%=mo;
        for(int i=last[now];i;i=a[i].ne) if(a[i].dao!=fa) get_ans(a[i].dao,now);
    }
}t1,t2;
inline int read()
{
    char c=getchar();
    int x=0,f=1;
    while(!isdigit(c)) {if(c==-) f=-1;c=getchar();}
    while(isdigit(c)) {x=(x<<3)+(x<<1)+c-0;c=getchar();}
    return x*f;
}
inline int hapow(int ch,int ci)
{
    int x=ch,gs=1;
    while(ci)
    {
        if(ci&1) gs=(gs*x)%mo;
        x=(x*x)%mo;
        ci>>=1;
    }
    return gs;
}
signed main()
{
    gsn=read();ch[0]=1;
    for(int i=1;i<=gsn+2;i++) ch[i]=(ch[i-1]*cc)%mo;
    t1.n=gsn;t2.n=gsn+1;
    for(int i=1,x,y;i<=gsn-1;i++)
    {
        x=read();y=read();
        t1.lingjiebiao(x,y);
        t1.lingjiebiao(y,x);
    }
    for(int i=1,x,y;i<=gsn;i++)
    {
        x=read();y=read();
        t2.lingjiebiao(x,y);
        t2.lingjiebiao(y,x);
    }
    t1.get_f(1,0);t2.get_f(1,0);
    t1.get_ans(1,0);t2.get_ans(1,0);
    for(int i=1;i<=gsn;i++) 
    ma[t1.ans[i]]=1;
    for(int i=1;i<=gsn+1;i++)
    {
        if(t2.cnt[i]>1) continue;
        if(ma[(t2.ans[i]*hapow(gsn+1,mo-2))%mo]==1)
        {
            cout<<i;
            return 0;
        }
    }
} 

独特的树叶

标签:输入   公式   sdi   整数   main   内存限制   +=   sig   void   

原文地址:https://www.cnblogs.com/betablewaloot/p/12146606.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!