题解 [51nod1673] 树有几多愁
解析
这题思路挺秒啊.
本麻瓜终于找了道好题了(还成功把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] 树有几多愁的更多相关文章
- [51nod1673]树有几多愁
lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输出最大烦 ...
- 51nod1673 树有几多愁 - 贪心策略 + 虚树 + 状压dp
传送门 题目大意: 给一颗重新编号,叶子节点的值定义为他到根节点编号的最小值,求所有叶子节点值的乘积的最大值. 题目分析: 为什么我觉得这道题最难的是贪心啊..首先要想到 在一条链上,深度大的编号要小 ...
- 刷题总结——树有几多愁(51nod1673 虚树+状压dp+贪心)
题目: lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输 ...
- 51nod 1673 树有几多愁
lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输出最大烦 ...
- 51nod 1673 树有几多愁(链表维护树形DP+状压DP)
题意 lyk有一棵树,它想给这棵树重标号. 重标号后,这棵树的所有叶子节点的值为它到根的路径上的编号最小的点的编号. 这棵树的烦恼值为所有叶子节点的值的乘积. lyk想让这棵树的烦恼值最大,你只需输出 ...
- 51nod 1673 树有几多愁——虚树+状压DP
题目:http://www.51nod.com/Challenge/Problem.html#!#problemId=1673 建一个虚树. 一种贪心的想法是把较小的值填到叶子上,这样一个小值限制到的 ...
- POJ2182题解——线段树
POJ2182题解——线段树 2019-12-20 by juruoOIer 1.线段树简介(来源:百度百科) 线段树是一种二叉搜索树,与区间树相似,它将一个区间划分成一些单元区间,每个单元区间对应线 ...
- Qtree3题解(树链剖分(伪)+线段树+set)
外话:最近洛谷加了好多好题啊...原题入口 这题好像是SPOJ的题,挺不错的.看没有题解还是来一篇... 题意: 很明显吧.. 题解: 我的做法十分的暴力:树链剖分(伪)+线段树+\(set\)... ...
- Qtree3题解(树链剖分+线段树+set)
外话:最近洛谷加了好多好题啊...原题入口 这题好像是SPOJ的题,挺不错的.看没有题解还是来一篇... 题意 很易懂吧.. 题解 我的做法十分的暴力:树链剖分(伪)+线段树+ std :: set ...
随机推荐
- [转帖]【Oracle】详解Oracle中NLS_LANG变量的使用
[Oracle]详解Oracle中NLS_LANG变量的使用 https://www.cnblogs.com/HDK2016/p/6880560.html NLS_LANG=LANGUAGE_TERR ...
- J.U.C之AQS介绍
从JDK1.5开始,引入了并发包java.util.concurrent(J.U.C),并发容器里的同步容器AQS(AbstractQueuedSynchronizer)是J.U.C的核心,AQS底层 ...
- gitlab安装指南(gitlab-ce-9.4.3-ce.0.el7.x86_64 centos7)
1,安装gitlab wget https://packages.gitlab.com/gitlab/gitlab-ce/packages/el/7/gitlab-ce-9.4.3-ce.0.el7. ...
- 【leetcode】74. 搜索二维矩阵
题目链接:传送门 题目描述 编写一个高效的算法来判断 m x n 矩阵中,是否存在一个目标值.该矩阵具有如下特性: 每行中的整数从左到右按升序排列. 每行的第一个整数大于前一行的最后一个整数. 示例 ...
- Codeforces 1189F. Array Beauty
传送门 首先可以注意到序列里面元素的顺序对答案是没有影响的,所以二话不说先排序再看看怎么搞 考虑枚举每种子序列可能产生的贡献并算一下产生这个贡献的子序列有多少 考虑设 $F(x)$ 表示选择的元素差值 ...
- [javascript]localStorage和sessionStorage区别
一.sessionStorage.localStorage.cookie可查看的位置,F12=>Application: 二.cookie .sessionStorage与localStorag ...
- ELinq学习一
ELinq安装:在Nuget控制台中输入:install-package ELinq一.ELinq与DLinq和EF的功能差异 二.数据库对照表 三.CRUD操作1.插入(Insert)(1)简单形式 ...
- 关于windows下编写的shell脚本在linux下无法运行报错问题
首先,你写的shell脚本必须是正确的, 其次,无法运行的原因:因为windows下的换行是两个字节,而你上传到linux,linux下换行是两个字节,所以编译的酒不正确的,导致无法 运行脚本, 这种 ...
- pthread 编程基础
Linux系统下的多线程遵循POSIX线程接口,称为pthread.编写Linux下的多线程程序,需要使用头文件pthread.h,连接时需要使用库libpthread.a.与vxworks上任务的概 ...
- Django 之form简单应用
form组件 参考链接:https://www.cnblogs.com/maple-shaw/articles/9537309.html form组件的作用: 1.自动生成input框 2.可以对数据 ...