树形DP(记忆化搜索) HYSBZ - 1509
题目链接:https://vjudge.net/problem/HYSBZ-1509
我参考的证明的论文:8.陈瑜希《多角度思考 创造性思维》_百度文库 https://wenku.baidu.com/view/f3b19d0b79563c1ec5da710e.html
第一道树形DP的题目,这里主要是寻找每一个点到其他点的距离里面最大的三个值,做两次记忆化搜索(DFS),第一次搜索求以每个点为根节点的子树(假设点1是整棵树的根节点)到它的所有儿子节点的距离里面最大的三个值(用数组mx1,mx2,mx3储存),第二次搜索是计算以每个点为子树,然后这个根节点到子树之外的所有点的距离里面的最大值(用数组f储存)。假设以点a为分叉点,那么点a的父亲到点a只有一条路,所以只要用f数组存一个子树外的最大值,剩下的两个就是子树里面的最大的两个距离,结果就是a+2*b+c,看不懂的话就看上面推荐的论文吧,我比较菜。
代码:(我可是加了注释的良心人)
#include<iostream>
#include<cstring>
#include<algorithm>
#include<queue>
#include<map>
#include<stack>
#include<cmath>
#include<vector>
#include<set>
#include<cstdio>
#include<string>
#include<deque>
using namespace std;
typedef long long LL;
#define eps 1e-8
#define INF 0x3f3f3f3f
#define maxn 200005
/*struct point{
int u,w;
};
bool operator <(const point &s1,const point &s2)
{
if(s1.w!=s2.w)
return s1.w>s2.w;
else
return s1.u>s2.u;
}*/
struct node{
int v,next;
LL w;
}edge[maxn<<];
int head[maxn];
int n,m,k,t,cnt;
LL mx1[maxn],mx2[maxn],mx3[maxn],f[maxn],ans;//mx1[i]代表点i为根到所有儿子距离的最大值
//f[i]表示点i到点i为根的子树外的所有点的距离最大值
bool vis[maxn];
void DFS(int u){//求出以u为根的子树中,点u到它的儿子节点的所有距离里面的最大三个
vis[u]=true;
for(int i=head[u];i!=-;i=edge[i].next){
int v=edge[i].v;
LL w=edge[i].w;
if(!vis[v]){
DFS(v);
mx3[u]=max(mx3[u],mx1[v]+w);//这里为什么这么写需要自己推敲一下
if(mx3[u]>mx2[u]) swap(mx2[u],mx3[u]);
if(mx2[u]>mx1[u]) swap(mx1[u],mx2[u]);
}
}
}
void DFS1(int u){//往父节点方向找子树之外的点到点u的最大距离
vis[u]=true;
for(int i=head[u];i!=-;i=edge[i].next){
int v=edge[i].v;
LL w=edge[i].w;
if(!vis[v]){
f[v]=f[u]+w;
if(mx1[u]==mx1[v]+w)//离点u距离最大的儿子节点在以点v为根节点的子树里面
f[v]=max(f[v],mx2[u]+w);
else
f[v]=max(f[v],mx1[u]+w);
DFS1(v);
}
}
}
void cal(LL a,LL b,LL c){
if(a<b) swap(a,b);//排序
if(a<c) swap(a,c);
if(b<c) swap(b,c);
ans=max(ans,a+*b+c);
}
void init(){
memset(head,-,sizeof(head));
memset(mx1,,sizeof(mx1));
memset(mx2,,sizeof(mx2));
memset(mx3,,sizeof(mx3));
memset(f,,sizeof(f));
cnt=ans=;
}
void add(int u,int v,LL w){
edge[++cnt].v=v;
edge[cnt].w=w;
edge[cnt].next=head[u];
head[u]=cnt;
}
int main()
{
scanf("%d%d",&n,&m);
init();
int u,v;
LL w;
for(int i=;i<=m;i++){
scanf("%d%d%lld",&u,&v,&w);
add(u,v,w);
add(v,u,w);
}
memset(vis,false,sizeof(vis));
DFS();
memset(vis,false,sizeof(vis));
DFS1();
for(int i=;i<=n;i++){//以每个点为分叉点的情况都算一遍
if(f[i]>mx3[i])
cal(mx1[i],mx2[i],f[i]);
else
cal(mx1[i],mx2[i],mx3[i]);
}
printf("%lld\n",ans);
return ;
}
树形DP(记忆化搜索) HYSBZ - 1509的更多相关文章
- 【bzoj5123】[Lydsy12月赛]线段树的匹配 树形dp+记忆化搜索
题目描述 求一棵 $[1,n]$ 的线段树的最大匹配数目与方案数. $n\le 10^{18}$ 题解 树形dp+记忆化搜索 设 $f[l][r]$ 表示根节点为 $[l,r]$ 的线段树,匹配选择根 ...
- [题解](树形dp/记忆化搜索)luogu_P1040_加分二叉树
树形dp/记忆化搜索 首先可以看出树形dp,因为第一个问题并不需要知道子树的样子, 然而第二个输出前序遍历,必须知道每个子树的根节点,需要在树形dp过程中记录,递归输出 那么如何求最大加分树——根据中 ...
- 刷题总结——二叉苹果树(ssoj树形dp+记忆化搜索)
题目: 题目背景 URAL:http://acm.timus.ru/problem.aspx?space=1&num=1018 题目描述 有一棵苹果树,如果树枝有分叉,一定是分 2 叉(就是说 ...
- 刷题总结——选课(ssoj树形dp+记忆化搜索+多叉树转二叉树)
题目: 题目描述 学校实行学分制.每门的必修课都有固定的学分,同时还必须获得相应的选修课程学分.学校开设了 N(N<300)门的选修课程,每个学生可选课程的数量 M 是给定的.学生选修了这M门课 ...
- 加分二叉树 vijos1991 NOIP2003第三题 区间DP/树形DP/记忆化搜索
描述 设一个n个节点的二叉树tree的中序遍历为(l,2,3,-,n),其中数字1,2,3,-,n为节点编号.每个节点都有一个分数(均为正整数),记第i个节点的分数为di,tree及它的每个子树都有一 ...
- 【BZOJ】1415 [Noi2005]聪聪和可可 期望DP+记忆化搜索
[题意]给定无向图,聪聪和可可各自位于一点,可可每单位时间随机向周围走一步或停留,聪聪每单位时间追两步(先走),问追到可可的期望时间.n<=1000. [算法]期望DP+记忆化搜索 [题解]首先 ...
- poj1664 dp记忆化搜索
http://poj.org/problem?id=1664 Description 把M个相同的苹果放在N个相同的盘子里,同意有的盘子空着不放,问共同拥有多少种不同的分法?(用K表示)5.1.1和1 ...
- 状压DP+记忆化搜索 UVA 1252 Twenty Questions
题目传送门 /* 题意:给出一系列的01字符串,问最少要问几个问题(列)能把它们区分出来 状态DP+记忆化搜索:dp[s1][s2]表示问题集合为s1.答案对错集合为s2时,还要问几次才能区分出来 若 ...
- ACM International Collegiate Programming Contest, Tishreen Collegiate Programming Contest (2017)- K. Poor Ramzi -dp+记忆化搜索
ACM International Collegiate Programming Contest, Tishreen Collegiate Programming Contest (2017)- K. ...
- POJ 1088 DP=记忆化搜索
话说DP=记忆化搜索这句话真不是虚的. 面对这道题目,题意很简单,但是DP的时候,方向分为四个,这个时候用递推就好难写了,你很难得到当前状态的前一个真实状态,这个时候记忆化搜索就派上用场啦! 通过对四 ...
随机推荐
- python 函数参数 *a **kw
f(name='a') name >>> def f(*a,**kw): print a for i in kw: print i >>> f([1,2],n='a ...
- CRC16-CCITT C语言代码
代码如下,使用空间换时间的方法 #define CRC16_CCITT_SEED 0xFFFF // 该位称为预置值,使用人工算法(长除法)时 需要将除数多项式先与该与职位 异或 ,才能得到最后的除数 ...
- VC 字符串转化和分割
原文:点击这里. 备忘:为了适用于Unicode环境,要养成使用_T()宏的习惯 1.格式化字符串 CString s;s.Format(_T("The num is %d."), ...
- 机器学习进阶-图像形态学操作-梯度运算 cv2.GRADIENT(梯度运算-膨胀图像-腐蚀后的图像)
1.op = cv2.GRADIENT 用于梯度运算-膨胀图像-腐蚀后的图像 梯度运算:表示的是将膨胀以后的图像 - 腐蚀后的图像,获得了最终的边缘轮廓 代码: 第一步:读取pie图片 第二步:进行腐 ...
- Excel和Word 简易工具类,JEasyPoi 2.1.7 版本发布
JEasyPOI 简介 EasyPOI 功能如同名字easy,追求的就是简易,让一个没接触过poi的人员,可以傻瓜化的快速实现Excel导入导出.Word模板导出,可以仅仅5行代码就可以完成Excel ...
- APP-1-相关介绍及资料
一年前研究了下MUI框架,也做了一些简单的功能,将整个过程整理下.. 1.Hbuilder官网 http://www.dcloud.io/ 2.MUI前端框架 http://www.dcloud.io ...
- 26个Jquery使用小技巧(转)
下面列出了一些Jquery使用技巧.比如有禁止右键点击.隐藏搜索文本框文字.在新窗口中打开链接.检测浏览器.预加载图片.页面样式切换.所有列等 高.动态控制页面字体大小.获得鼠标指针的X值Y值.验证元 ...
- c++复习:STL之容器
1 STL的string 1 String概念 string是STL的字符串类型,通常用来表示字符串.而在使用string之前,字符串通常是用char*表示的.string与char*都可以用来表示字 ...
- Window环境下Python和Django的安装,以及项目的创建
1.首先我们要下载python和Django,他们的下载地址如下 python地址:https://www.python.org/ Django地址: https://www.djangoproje ...
- HG255D刷机OPENWRT笔记
1.按网上的办法如下:(http://www.right.com.cn/forum/thread-143721-1-1.html) 自已编译了OPENWRT,然后拆开外壳接上TTL线,通电启动 然后用 ...