热门标签 | 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
*/


推荐阅读
  • 洛谷 P4009 汽车加油行驶问题 解析
    探讨了经典算法题目——汽车加油行驶问题,通过网络流和费用流的视角,深入解析了该问题的解决方案。本文将详细阐述如何利用最短路径算法解决这一问题,并提供详细的代码实现。 ... [详细]
  • 本文详细介绍了如何在循环双链表的指定位置插入新元素的方法,包括必要的步骤和代码示例。 ... [详细]
  • 本问题涉及在给定的无向图中寻找一个至少包含三个节点的环,该环上的节点不重复,并且环上所有边的长度之和最小。目标是找到并输出这个最小环的具体方案。 ... [详细]
  • c语言二元插值,二维线性插值c语言
    c语言二元插值,二维线性插值c语言 ... [详细]
  • 实现系统调用
    实现系统调用一、实验环境​本次操作还是基于上次编译Linux0.11内核的实验环境进行操作。环境如下:二、实验目标​通过对上述实验原理的认识,相信 ... [详细]
  • 本文探讨了如何高效地计算数组中和为2的幂的偶对数量,提供了从基础到优化的方法。 ... [详细]
  • 编译原理中的语法分析方法探讨
    本文探讨了在编译原理课程中遇到的复杂文法问题,特别是当使用SLR(1)文法时遇到的多重规约与移进冲突。文章讨论了可能的解决策略,包括递归下降解析、运算符优先级解析等,并提供了相关示例。 ... [详细]
  • 本文详细介绍了C++中的构造函数,包括其定义、特点以及如何通过构造函数进行对象的初始化。此外,还探讨了转换构造函数的概念及其在不同情境下的应用,以及如何避免不必要的隐式类型转换。 ... [详细]
  • 本文通过分析一个具体的案例,探讨了64位Linux系统对32位应用程序的兼容性问题。案例涉及OpenVPN客户端在64位系统上的异常行为,通过逐步排查和代码测试,最终定位到了与TUN/TAP设备相关的系统调用兼容性问题。 ... [详细]
  • 本题要求计算一组正整数的最小公倍数(LCM)。输入包括多组测试数据,每组数据首先给出一个正整数n,随后是n个正整数。 ... [详细]
  • linux网络子系统分析(二)—— 协议栈分层框架的建立
    目录一、综述二、INET的初始化2.1INET接口注册2.2抽象实体的建立2.3代码细节分析2.3.1socket参数三、其他协议3.1PF_PACKET3.2P ... [详细]
  • 本文档介绍了如何使用ESP32开发板在STA模式下实现与TCP服务器的通信,包括环境搭建、代码解析及实验步骤。 ... [详细]
  • 使用QT构建基础串口辅助工具
    本文详细介绍了如何利用QT框架创建一个简易的串口助手应用程序,包括项目的建立、界面设计与编程实现、运行测试以及最终的应用程序打包。 ... [详细]
  • 线段树详解与实现
    本文详细介绍了线段树的基本概念及其在编程竞赛中的应用,并提供了一个具体的线段树实现代码示例。 ... [详细]
  • 本文介绍了如何利用X_CORBA实现远程对象调用,并通过多个示例程序展示了其功能与应用,包括基础的Hello World示例、文件传输工具以及一个完整的聊天系统。 ... [详细]
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社区 版权所有