热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

2019MultiUniversityTrainingContest31011Squrirrel——树形DP

Thisway题意:给你一棵树,路径上都有权值,你现在能将一条边权变为0,问你从某个节点出发到最远的叶子结点的路径最短是

This way

题意:

给你一棵树,路径上都有权值,你现在能将一条边权变为0,问你从某个节点出发到最远的叶子结点的路径最短是多少。

题解:

这道题乍一看不难,但是它的情况很多,所以debug了很久才过。首先我们处理出每个节点距离它子树的叶子结点的最远,第二远,第三远的距离,再用DP做出每个节点删掉一条边后到叶子结点最远,第二远的距离。之后再用一个dfs做答案,首先参数需要传:
1.从父亲传下来的,不包括这颗子树的最远未删边距离,2.从父亲传下来的,不包括这颗子树的最优最远删边距离。
那么这个节点的答案就是
min{max{父亲未删边,这棵树删边最大,这棵树未删边第二大},max{父亲删边最优最大,这棵树未删边最大}}
为什么,因为不确定删边最大和未删边第二大的关系,所以需要判一下。
先说一下重要数组的意义:
no_mx:表示未删边最大
no_smx:表示未删边第二大
del_mx:表示删边最大
no_:表示父亲传下来的未删边最大
del_:表示父亲传下来的删边最大
那么之后麻烦的就来了,首先对于他的某一个儿子节点,
1.如果他是这个点的未删边最大的儿子节点:
1.1如果他是这个点的删边最大儿子节点:
1.1.1他就会有no_smx,no_tmx,del_smx,del_tmx四种可以传下去的,那么我们肯定是删掉no_smx,那么也就是说在no_tmx和del_smx中选择最大的
1.2如果他是这个点的第二或者其它儿子节点:
1.2.1他就会有no_smx,no_tmx,del_mx,del_tmx四种可以传下去的,那么我们不知道删掉no_smx还是del_mx,那么我们就在这两个中取一个最小的,在之后与no_tmx取个最大的
2.如果他是这个节点的未删边第二:
2.1如果它是这个点的删边最大儿子节点:
2.1.1他就会有no_mx,no_tmx,del_mx,del_tmx,那么我们肯定是删掉no_mx,所以在del_smx与no_tmx中取个最大的
2.2如果它是这个节点的删边第二大儿子节点或者其它:
2.2.1它就会有no_mx,no_tmx,del_mx,del_tmx四种,那么我们删掉no_mx,所以在del_mx,与no_tmx中取个最大的
3.如果它是这个节点的未删边第三或者其它节点:
3.1如果它是这个点的删边最大儿子节点:
3.1.1它就会有no_mx,no_smx,del_smx,del_tmx四种,那么我们删掉no_mx,在no_smx,del_smx中取最大的
3.2如果它是这个点的删边第二或者其它儿子节点:
3.2.1它就会有no_mx,no_smx,del_mx,del_tmx四种,那么我们删掉no_mx,在no_smx,del_mx中取最大的

情况数有点多,令人绝望的代码。

#include
using namespace std;
const int N=2e6+5;
int dp[N][2],no_mx[N],no_smx[N],no_tmx[N],del_mx[N],del_smx[N],del_tmx[N],ans[N];
struct node
{int to,next,w;
}e[N*2];
int cnt,head[N];
void add(int x,int y,int w)
{e[cnt].to=y;e[cnt].next=head[x];e[cnt].w=w;head[x]=cnt++;
}
void dfs(int x,int fa)
{for(int i=head[x];~i;i=e[i].next){int ne=e[i].to,w=e[i].w;if(ne==fa)continue;dfs(ne,x);if(no_mx[ne]+w>no_mx[x])no_tmx[x]=no_smx[x],no_smx[x]=no_mx[x],no_mx[x]=no_mx[ne]+w;else if(no_mx[ne]+w>no_smx[x])no_tmx[x]=no_smx[x],no_smx[x]=no_mx[ne]+w;else if(no_mx[ne]+w>no_tmx[x])no_tmx[x]=no_mx[ne]+w;int d=min(dp[ne][1]+w,dp[ne][0]);dp[x][1]=min(max(dp[x][0],d),max(dp[x][1],dp[ne][0]+w));dp[x][0]=max(dp[x][0],dp[ne][0]+w);if(d>del_mx[x])del_tmx[x]=del_smx[x],del_smx[x]=del_mx[x],del_mx[x]=d;else if(d>del_smx[x])del_tmx[x]=del_smx[x],del_smx[x]=d;else if(d>del_tmx[x])del_tmx[x]=d;}
}
int p,val;
void fans(int x,int fa,int no_,int del_)
{ans[x]=min(max(no_,max(del_mx[x],no_smx[x])),max(del_,no_mx[x]));if(ans[x]x)p=x;for(int i=head[x];~i;i=e[i].next){int ne=e[i].to,w=e[i].w;if(ne==fa)continue;int nv=no_+w,nd;if(no_mx[ne]+w==no_mx[x]){nv=max(nv,no_smx[x]+w);nd=min(max(no_,no_smx[x]),max(del_+w,no_smx[x]+w));//去掉这条边,父亲传来的值if(min(dp[ne][1]+w,dp[ne][0])==del_mx[x])nd=min(nd,max(del_smx[x]+w,max(no_+w,no_tmx[x]+w)));//去掉其它儿子边elsend=min(nd,max(no_+w,max(no_tmx[x]+w,min(del_mx[x]+w,no_smx[x]+w))));}else if(no_mx[ne]+w==no_smx[x]){nv=max(nv,no_mx[x]+w);nd=min(max(no_,no_mx[x]),max(del_+w,no_mx[x]+w));if(min(dp[ne][1]+w,dp[ne][0])==del_mx[x])nd=min(nd,max(del_smx[x]+w,max(no_+w,no_tmx[x]+w)));elsend=min(nd,max(no_+w,max(no_tmx[x]+w,del_mx[x]+w)));}else{nv=max(nv,no_mx[x]+w);nd=min(max(no_,no_mx[x]),max(del_+w,no_mx[x]+w));if(min(dp[ne][1]+w,dp[ne][0])==del_mx[x])nd=min(nd,max(no_+w,max(no_smx[x]+w,del_smx[x]+w)));elsend=min(nd,max(no_+w,max(no_smx[x]+w,del_mx[x]+w)));}fans(ne,x,nv,nd);}
}
int main()
{//freopen("1.in","r",stdin);//freopen("out.txt","w",stdout);int t;scanf("%d",&t);while(t--){int n;scanf("%d",&n);memset(head,-1,sizeof(head));cnt&#61;0;val&#61;1e9;for(int i&#61;1;i<&#61;n;i&#43;&#43;)dp[i][0]&#61;dp[i][1]&#61;no_mx[i]&#61;no_smx[i]&#61;no_tmx[i]&#61;del_mx[i]&#61;del_smx[i]&#61;del_tmx[i]&#61;ans[i]&#61;0;int x,y,w;for(int i&#61;1;i}
/*
1
5
1 2 1
2 3 1
3 4 5
3 5 11
5
1 5 1
1 2 1
2 3 2
3 4 11
10
2 1 155
3 1 29
4 2 33
5 2 178
6 3 42
7 2 56
8 6 189
9 4 99
10 5 63
*/


推荐阅读
  • P2765魔术球问题这道题的思路实在是太罕见了,所以发一篇blog从某一新放入的球开始看起1.放入原来的柱子上2.放入新的柱子并将每个点进行拆点࿰ ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • 本文介绍了P1651题目的描述和要求,以及计算能搭建的塔的最大高度的方法。通过动态规划和状压技术,将问题转化为求解差值的问题,并定义了相应的状态。最终得出了计算最大高度的解法。 ... [详细]
  • 本文介绍了使用哈夫曼树实现文件压缩和解压的方法。首先对数据结构课程设计中的代码进行了分析,包括使用时间调用、常量定义和统计文件中各个字符时相关的结构体。然后讨论了哈夫曼树的实现原理和算法。最后介绍了文件压缩和解压的具体步骤,包括字符统计、构建哈夫曼树、生成编码表、编码和解码过程。通过实例演示了文件压缩和解压的效果。本文的内容对于理解哈夫曼树的实现原理和应用具有一定的参考价值。 ... [详细]
  • 本文介绍了一道经典的状态压缩题目——关灯问题2,并提供了解决该问题的算法思路。通过使用二进制表示灯的状态,并枚举所有可能的状态,可以求解出最少按按钮的次数,从而将所有灯关掉。本文还对状压和位运算进行了解释,并指出了该方法的适用性和局限性。 ... [详细]
  • 本文介绍了设计师伊振华受邀参与沈阳市智慧城市运行管理中心项目的整体设计,并以数字赋能和创新驱动高质量发展的理念,建设了集成、智慧、高效的一体化城市综合管理平台,促进了城市的数字化转型。该中心被称为当代城市的智能心脏,为沈阳市的智慧城市建设做出了重要贡献。 ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 本文讨论了使用差分约束系统求解House Man跳跃问题的思路与方法。给定一组不同高度,要求从最低点跳跃到最高点,每次跳跃的距离不超过D,并且不能改变给定的顺序。通过建立差分约束系统,将问题转化为图的建立和查询距离的问题。文章详细介绍了建立约束条件的方法,并使用SPFA算法判环并输出结果。同时还讨论了建边方向和跳跃顺序的关系。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • JDK源码学习之HashTable(附带面试题)的学习笔记
    本文介绍了JDK源码学习之HashTable(附带面试题)的学习笔记,包括HashTable的定义、数据类型、与HashMap的关系和区别。文章提供了干货,并附带了其他相关主题的学习笔记。 ... [详细]
  • 本文由编程笔记#小编整理,主要介绍了关于数论相关的知识,包括数论的算法和百度百科的链接。文章还介绍了欧几里得算法、辗转相除法、gcd、lcm和扩展欧几里得算法的使用方法。此外,文章还提到了数论在求解不定方程、模线性方程和乘法逆元方面的应用。摘要长度:184字。 ... [详细]
  • 本文讲述了CodeForces1016C题目的解法。文章首先介绍了一种错误的理解,然后给出了正确的解法。其中,当位于一个角上时,有两种选择,一种是先一直走一行再返回来走,另一种是走到这一列的另一行上然后再往右走一列。作者给出了两种解法,一种是直接计算,一种是动态规划。最后,取两种解法的最优解作为答案。文章附上了源代码。 ... [详细]
  • 796.[APIO2012]派遣在一个忍者的帮派里,一些忍者们被选中派遣给顾客,然后依据自己的工作获取报偿。在这个帮派里,有一名忍者被称之为Master。 ... [详细]
  • 题目描述Takuru是一名情报强者,所以他想利用他强大的情报搜集能力来当中间商赚差价。Takuru的计划是让Hinae帮他去市场上买一个商品,然后再以另一个价格卖掉它。Takur ... [详细]
  • DescriptionclickmeSolution套路的状压期望DP题。。。考虑倒退期望:设fi,jrolepresentationstyleposi ... [详细]
author-avatar
周同学天天爬十楼7634
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有