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

题解 [51nod1673] 树有几多愁

时间:2019-08-28 11:14:38      阅读:91      评论:0      收藏:0      [点我收藏+]

标签:stdout   lin   cst   hal   mod   ios   想法   long   node   

题面

解析

这题思路挺秒啊.

本麻瓜终于找了道好题了(还成功把ztlztl大仙拖下水了)

看到叶子节点数<=20就应该是状压啊.

然而DP要怎么写啊?

首先,考虑到编号肯定是从下往上一次增大的,

另外,对于没有分支的一条链,它的编号应该是连续的.

并且一种类似于贪心的想法就是一个点\(u\)被编号时它的子树一定被编号完了.

所以这也像是一个类似于拓扑序的东西.

先建一棵虚树(因为叶子节点只有20有很多没用的点),边权设为这条链上不在虚树上的点数.

设状态\(i\)表示状压后集合\(i\)中的点的编号已经确定了.

那么我们可以把所有已经编号了的点数\(cnt\)求出来,

然后枚举没在点集中的叶子节点,它的编号就应该是\(cnt+1\),再更新答案就行了.

因为取模后无法比较大小所以我们可以另外开一个\(double\)数组来比较.

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#define int long long
#define fre(x) freopen(x".in","r",stdin),freopen(x".out","w",stdout)
using namespace std;

inline int read(){
    int sum=0,f=1;char ch=getchar();
    while(ch>'9' || ch<'0'){if(ch=='-')f=-1;ch=getchar();}
    while(ch>='0' && ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
    return f*sum;
}

const int N=100005;
const int M=(1<<20)+5;
const int Mod=1000000007;
struct edge{int to,next,w;}e[N<<1];
struct node{int size,dep,fa,son,top,dfn,is;}a[N];
int n;
int head[N],cnt=0,tp;
int q[N],sta[N],tot,top;
int f[M],s[N];double dp[M];
int que[N],qq[N],tt;

inline void add(int x,int y,int w){
    e[++cnt]=(edge){head[x],y,w-1};head[x]=cnt;
}

inline void dfs(int x,int fa){
    a[x].dep++;a[x].fa=fa;
    a[x].size=a[x].is=1;a[x].dfn=++tp;
    for(int i=head[x];i;i=e[i].to){
        int k=e[i].next;if(k==fa) continue;
        a[k].dep=a[x].dep+e[i].w;
        dfs(k,x);a[x].size+=e[i].w+a[k].size;
        if(a[k].size>a[a[x].son].size) a[x].son=k;
        a[x].is=0;
    }
}

inline void dfs2(int x,int top){
    a[x].top=top;
    if(a[x].son) dfs2(a[x].son,top);
    for(int i=head[x];i;i=e[i].to){
        int k=e[i].next;
        if(k==a[x].son||k==a[x].fa) continue;
        dfs2(k,k);
    }
}

inline int lca(int x,int y){
    while(a[x].top!=a[y].top){
        if(a[a[x].top].dep<a[a[y].top].dep) swap(x,y);
        x=a[a[x].top].fa;
    }
    if(a[x].dep>a[y].dep) swap(x,y);
    return x;
}//树剖求lca

inline bool cmp(int x,int y){return a[x].dfn<a[y].dfn;}

signed main(){
    n=read();
    for(int i=1;i<n;i++){int x=read(),y=read();add(x,y,1);add(y,x,1);}
    dfs(1,0);dfs2(1,1);
    for(int i=1;i<=n;i++) if(a[i].is) q[++tot]=i;
    sort(q+1,q+tot+1,cmp);if(q[1]!=1) sta[++top]=1;
    memset(head,0,sizeof(head));cnt=0;
    //*******
    for(int i=1;i<=tot;i++){
        if(!top){sta[++top]=q[i];continue;}
        int p=lca(sta[top],q[i]);
        while(top>1&&a[sta[top-1]].dep>=a[p].dep)
        {add(sta[top-1],sta[top],a[sta[top]].dep-a[sta[top-1]].dep);top--;}
        if(sta[top]!=p) add(p,sta[top],a[sta[top]].dep-a[p].dep),sta[top]=p;
        sta[++top]=q[i];
    }
    while(top>1) add(sta[top-1],sta[top],a[sta[top]].dep-a[sta[top-1]].dep),top--;
    //*******建虚树
    a[1].dep=0;dfs(1,0);int lim=1<<tot;
    f[0]=dp[0]=1;
    for(int i=0;i<lim;i++){
        for(int j=1;j<=tot;j++) if((i&(1<<(j-1)))) s[q[j]]=1;
        int ret=0,l=1,r=0;tt=0;
        for(int j=1;j<=tot;j++) if(s[q[j]]) que[++r]=q[j];
        while(l<=r){
            int x=que[l];l++;qq[++tt]=x;
            ret+=a[x].dep-a[a[x].fa].dep;
            if(!a[x].fa) continue;qq[++tt]=a[x].fa;
            s[a[x].fa]+=s[x]+a[x].dep-a[a[x].fa].dep-1;
            if(s[a[x].fa]==a[a[x].fa].size-1) s[a[x].fa]++,que[++r]=a[x].fa;
        }//像拓扑序一样统计数量       
        for(int i=1;i<=tt;i++) s[qq[i]]=0;//清空s数组(之前用memsetT得一脸懵逼)
        ret++;
        for(int j=1;j<=tot;j++){
            if((i&(1<<(j-1)))) continue;
            int k=i|(1<<(j-1));
            if(dp[k]<dp[i]*ret) f[k]=f[i]*ret%Mod,dp[k]=dp[i]*ret;
        }
    }
    printf("%lld\n",f[lim-1]);
    return 0;
}

题解 [51nod1673] 树有几多愁

标签:stdout   lin   cst   hal   mod   ios   想法   long   node   

原文地址:https://www.cnblogs.com/zsq259/p/11422375.html

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