洛谷P4438 道路 [HNOI/AHOI2018] 树形dp
正解:树形dp
解题报告:
昂首先看懂题目趴QwQ大概就是说有棵满二叉树,有n个叶子节点(乡村)和n-1个非叶子节点,然后这棵树的每个节点有三个属性abc,对每个非叶子节点可以从与子节点的两条连边中选一条标记
然后对每个叶子节点i,设它到根节点经过了x条麻油被标记的左连边,y条麻油被标记的右连边,那它的贡献就会是ci*(ai-x)*(bi-y)
然后就考虑dp鸭
设f[x][i][j]:对于节点x,到达根的路径上有i条麻油标记的左连边,j条麻油标记的右连边
然后初始化是对叶子节点,直接枚举i和j
然后转移就是对非叶子节点,可以标记左连边可以标记右连边,就f[x][i][j]=min(f[x.ls][i][j]+f[x.rs][i][j+1],f[x.ls][i+1][j]+f[x.rs][i][j])
答案就是f[1][0][0]嘛
然后这题就做完辣!看起来好简单的样子
另外就是,这道题如果开的是f[N][40][40]显然是会爆空间的QAQ所以要想下怎么卡空间QAQ
然后仔细思考一下,因为这是一棵树,而且转移一定是从根dfs到儿子节点的,所以当两个儿子节点都访问完之后一定不会再回来辣
所以可以用,记录dfn的方式,这里的dfn并不是真正每次都++的那种(不然完全麻油省空间鸭QAQ),它是指的dfn[x.ls]=dfn[x]+1,dfn[x.rs]=dfn[x]+2,显然这样是可以保证正确性的,然后这样就能保障f的第一维<=2*40+1,这样就肯定不会爆空间辣!
然后记得它其实是有2*n-1个节点的,所以dfn序的那个数组要开两倍N,,,
虽然我jio得不会有人和我一样犯这种sd错误辣?
而且我居然RE了两三次才发现是这个问题,,,吃枣药丸TT
然后放下代码就溜辣!
#include<bits/stdc++.h>
using namespace std;
#define il inline
#define rg register
#define ll long long
#define gc getchar()
#define rp(i,x,y) for(rg int i=x;i<=y;++i)
#define my(i,x,y) for(rg int i=x;i>=y;--i) const int N=+;
int n,m,dfn[N];
long long f[][][];
struct node{int g,t;}nod[N];
struct leaf{int a,b,c;}lf[N]; il int read()
{
rg char ch=gc;rg int x=;rg bool y=;
while(ch!='-' && (ch<'' || ch>''))ch=gc;
if(ch=='-')ch=gc,y=;
while(''<=ch && ch<='')x=(x<<)+(x<<)+(ch^''),ch=gc;
return y?x:-x;
}
il void dfs(int x,int nw,int gl,int tl)
{
dfn[x]=nw;
if(x>=n){rp(i,,gl)rp(j,,tl)f[dfn[x]][i][j]=1ll*lf[x].c*1ll*(lf[x].a+i)*(lf[x].b+j);return;}
dfs(nod[x].g,nw+,gl+,tl);dfs(nod[x].t,nw+,gl,tl+);
rp(i,,gl)rp(j,,tl)f[dfn[x]][i][j]=min(f[nw+][i+][j]+f[nw+][i][j],f[nw+][i][j]+f[nw+][i][j+]);
} int main()
{
// freopen("dl.in","r",stdin);freopen("dl.out","w",stdout);
n=read();
rp(i,,n-){int s=read(),t=read();if(s>)nod[i].g=s;else nod[i].g=n-s-;if(t>)nod[i].t=t;else nod[i].t=n-t-;}rp(i,,n-)lf[i+n].a=read(),lf[i+n].b=read(),lf[i+n].c=read();
dfs(,,,);printf("%lld\n",f[][][]);
return ;
}
嗷对了这题还有一个解法来着QwQ
只是不管是时间上还是空间上都不太优秀,而且非常好想,就很适合我这种菜菜,然后居然能过,实在是很友好了TT
就直接无脑记搜就好,放下代码就能get了应该QAQ?
#include<bits/stdc++.h>
using namespace std;
#define il inline
#define rg register
#define ll long long
#define gc getchar()
#define rp(i,x,y) for(rg ll i=x;i<=y;++i)
#define my(i,x,y) for(rg ll i=x;i>=y;--i) const ll N=+;
ll n,m,f[N][][];
struct node{ll ls,rs;}nod[N];
struct leaf{ll a,b,c;}lf[N]; il ll read()
{
rg char ch=gc;rg ll x=;rg bool y=;
while(ch!='-' && (ch<'' || ch>''))ch=gc;
if(ch=='-')ch=gc,y=;
while(''<=ch && ch<='')x=(x<<)+(x<<)+(ch^''),ch=gc;
return y?x:-x;
}
il ll dfs(ll x,ll i,ll j)
{
if(x>=n)return lf[x-n].c*(lf[x-n].a+i)*(lf[x-n].b+j);if(f[x][i][j]!=f[n+][][])return f[x][i][j];
return f[x][i][j]=min(dfs(nod[x].ls,i,j)+dfs(nod[x].rs,i,j+),dfs(nod[x].ls,i+,j)+dfs(nod[x].rs,i,j));
} int main()
{
// freopen("dl.in","r",stdin);
n=read();rp(i,,n-){ll s=read(),t=read();if(s>)nod[i].ls=s;else nod[i].ls=n-s-;if(t>)nod[i].rs=t;else nod[i].rs=n-t-;}rp(i,,n-)lf[i].a=read(),lf[i].b=read(),lf[i].c=read();
memset(f,,sizeof(f));printf("%lld\n",dfs(,,));
return ;
}
洛谷P4438 道路 [HNOI/AHOI2018] 树形dp的更多相关文章
- 洛谷 P3177 [HAOI2015]树上染色 树形DP
洛谷 P3177 [HAOI2015]树上染色 树形DP 题目描述 有一棵点数为 \(n\) 的树,树边有边权.给你一个在 \(0 \sim n\)之内的正整数 \(k\) ,你要在这棵树中选择 \( ...
- 洛谷P4426 毒瘤 [HNOI/AHOI2018] 虚树+树上dp
正解:虚树+树上dp 解题报告: 传送门! 首先解释一下题意趴,,,语文70pts选手已经开始看不懂题辣QAQ 大概就是个给一个图,求独立集方案,且保证图是联通的,边的数量最多只比点多10 首先思考如 ...
- 【题解】洛谷P1070 道路游戏(线性DP)
次元传送门:洛谷P1070 思路 一开始以为要用什么玄学优化 没想到O3就可以过了 我们只需要设f[i]为到时间i时的最多金币 需要倒着推回去 即当前值可以从某个点来 那么状态转移方程为: f[i]= ...
- 洛谷P1040 加分二叉树(树形dp)
加分二叉树 时间限制: 1 Sec 内存限制: 125 MB提交: 11 解决: 7 题目描述 设一个n个节点的二叉树tree的中序遍历为(l,2,3,...,n),其中数字1,2,3,...,n ...
- 洛谷 P4201 设计路线 [NOI2008] 树形dp
正解:树形dp 解题报告: 大概是第一道NOI的题目?有点激动嘻嘻 然后先放个传送门 先大概港下这题的题意是啥qwq 大概就是给一棵树,然后可以选若干条链把链上的所有边的边权变成0,但是这些链不能有交 ...
- 洛谷 P3267 [JLOI2016/SHOI2016]侦察守卫(树形dp)
题面 luogu 题解 树形\(dp\) \(f[x][y]表示x的y层以下的所有点都已经覆盖完,还需要覆盖上面的y层的最小代价.\) \(g[x][y]表示x子树中所有点都已经覆盖完,并且x还能向上 ...
- 洛谷P1351 联合权值(树形dp)
题意 题目链接 Sol 一道很简单的树形dp,然而被我写的这么长 分别记录下距离为\(1/2\)的点数,权值和,最大值.以及相邻儿子之间的贡献. 树形dp一波.. #include<bits/s ...
- 洛谷P4099 [HEOI2013]SAO(树形dp)
传送门 HEOI的题好珂怕啊(各种意义上) 然后考虑树形dp,以大于为例 设$f[i][j]$表示$i$这个节点在子树中排名第$j$位时的总方案数(因为实际只与相对大小有关,与实际数值无关) 我们考虑 ...
- 洛谷 P1351 联合权值 —— 树形DP
题目:https://www.luogu.org/problemnew/show/P1351 树形DP,别忘了子树之间的情况(拐一下距离为2). 代码如下: #include<iostream& ...
随机推荐
- C++开发--在Visual Studio2013中使用boost::split()编译过程中出现error C4996
Visual Studio is being overly cautious. In debug mode, visual studio uses something called "Ch ...
- Linux下安装配置MySQL
一.删除原来的MySQL 在安装前要先确定系统是否已经安装了其他版本的MySQL,如已安装其他版本的MySQL,需先删除后再安装新版本. 1. 执行yum命令,删除MySQL的lib库,服务文件 yu ...
- git切换分支(自记)
git fetch git checkout feature/A4-page
- mybatis的selectOne和selectList没有数据返回时的问题
1.使用mybatis的selectList方法,如果数据表中没有数据返回,则返回空集合[ ],而不会返回null,这是mybatis作的封装 @Override public List<Con ...
- Qt下libusb-win32的使用方法
之前一直找不到适合WIN7下的Tiny6410的USB下载软件,正好这几天开始学习USB,所以打算自己写一个专门用于Tiny6410的WIN7下的USB下载软件. 发现了libusb这个库可以用作无驱 ...
- IOS设计模式第三篇之外观设计模式
外观设计模式: 这个外观设计模式提供了一个单独的接口给复杂的子系统.而不是暴露用户的一组类和API,你仅仅暴露一个简单的同一的API. 下面的图片解释这个概念: API的用户根本不知道后面系统的复杂性 ...
- Geeks面试题:Min Cost Path
Min Cost Path Given a cost matrix cost[][] and a position (m, n) in cost[][], write a function tha ...
- C# 给主程序签名及第三方dll强签名操作
1.给主程序添加签名 添加完成后会自动生成一个*.pfx文件. 2.给第三方程序添加强签名方法: 本文以WAPIWrapperCSharp.dll为例,使用vs Tools下的工具命令 ...
- Oracle 学习之exists
不相关子查询:子查询的查询条件不依赖于父查询的称为不相关子查询.相关子查询:子查询的查询条件依赖于外层父查询的某个属性值的称为相关子查询,带EXISTS 的子查询就是相关子查询EXISTS表示存在量词 ...
- [转]F5负载均衡算法及基本原理
原文:Intro to Load Balancing for Developers – The Algorithms 转载:http://blog.gesha.net/archives/205/ p ...