题面

解析

这题思路挺秒啊.

本麻瓜终于找了道好题了(还成功把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] 树有几多愁的更多相关文章

  1. [51nod1673]树有几多愁

    lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输出最大烦 ...

  2. 51nod1673 树有几多愁 - 贪心策略 + 虚树 + 状压dp

    传送门 题目大意: 给一颗重新编号,叶子节点的值定义为他到根节点编号的最小值,求所有叶子节点值的乘积的最大值. 题目分析: 为什么我觉得这道题最难的是贪心啊..首先要想到 在一条链上,深度大的编号要小 ...

  3. 刷题总结——树有几多愁(51nod1673 虚树+状压dp+贪心)

    题目: lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输 ...

  4. 51nod 1673 树有几多愁

    lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输出最大烦 ...

  5. 51nod 1673 树有几多愁(链表维护树形DP+状压DP)

    题意 lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输出 ...

  6. 51nod 1673 树有几多愁——虚树+状压DP

    题目:http://www.51nod.com/Challenge/Problem.html#!#problemId=1673 建一个虚树. 一种贪心的想法是把较小的值填到叶子上,这样一个小值限制到的 ...

  7. POJ2182题解——线段树

    POJ2182题解——线段树 2019-12-20 by juruoOIer 1.线段树简介(来源:百度百科) 线段树是一种二叉搜索树,与区间树相似,它将一个区间划分成一些单元区间,每个单元区间对应线 ...

  8. Qtree3题解(树链剖分(伪)+线段树+set)

    外话:最近洛谷加了好多好题啊...原题入口 这题好像是SPOJ的题,挺不错的.看没有题解还是来一篇... 题意: 很明显吧.. 题解: 我的做法十分的暴力:树链剖分(伪)+线段树+\(set\)... ...

  9. Qtree3题解(树链剖分+线段树+set)

    外话:最近洛谷加了好多好题啊...原题入口 这题好像是SPOJ的题,挺不错的.看没有题解还是来一篇... 题意 很易懂吧.. 题解 我的做法十分的暴力:树链剖分(伪)+线段树+ std :: set ...

随机推荐

  1. java源码--Vector和Stack

    一.Vector简介 1.1.Vector概述 通过API中可以知道: 1)Vector是一个可变化长度的数组 2)Vector增加长度通过的是capacity和capacityIncrement这两 ...

  2. 整体二分(模板二)动态区间第K大

    这才是更一般的二分写法--HDU5412 #define IOS ios_base::sync_with_stdio(0); cin.tie(0); #include <cstdio>// ...

  3. 如何理解归一化(Normalization)对于神经网络(深度学习)的帮助?

    如何理解归一化(Normalization)对于神经网络(深度学习)的帮助? 作者:知乎用户链接:https://www.zhihu.com/question/326034346/answer/730 ...

  4. MyEclipse eclipse console edit packageExplorer 颜色设置、个性化、常用设置

    下列教程的图片是在 myeclipse2014 破解版上进行,会有些许不同,仅供参考! 1 编辑区颜色设置 主题设置 豆沙绿设置 RGB 203 233 207 2 console 3主题选择 4 去 ...

  5. [JZOJ4307]喝喝喝--枚举

    [JZOJ4307]喝喝喝--枚举 题目链接 自行搜索 分析 我们需要找到所有不包含\((a_x,a_y),a_x \equiv k \mod a_y (x<y)\)这样的连续数对,转化一下变成 ...

  6. js重点——作用域——简单介绍(一)

    一.作用域 定义:在js中,作用域为变量,对象,函数可访问的一个范围. 分类:全局作用域和局部作用域 全局作用域:全局代表了整个文档document,变量或者函数在函数外面声明,那它的就是全局变量和全 ...

  7. Linux学习(一)-安装vm虚拟机以及如何在虚拟机上安装Centos系统

    (一)基本说明 学习Linux需要一个环境,我们需要创建一个虚拟机,然后在虚拟机上安装一个Centos系统来学习. 1)安装软件vm12; 2)通过vm12创建一个虚拟机空间; 3)在vm12创建好的 ...

  8. 微信小程序点击图片预览-wx.previewImage

    <view class='imgList'> <view class='imgList-li' wx:for='{{imgArr}}'> <image class='im ...

  9. 3.web开发入门知识

    /*web入门*/ /*互联网上常用的协议以及它的端口*/ http 80 http://localhost/    相当于    http://localhost:80/         http协 ...

  10. 运行tomcat7w.exe提示指定的服务未安装 解决办法

    一.问题重现点击bin下tomcat7w.exe出现如下提示:提示指定的服务未安装 二.原因分析tomcat7.exe和tomcat7w.exe要起作用必须先为这两个文件安装服务.其中tomcat7. ...