题目描述

几乎整个Byteland王国都被森林和河流所覆盖。小点的河汇聚到一起,形成了稍大点的河。就这样,所有的河水都汇聚并流进了一条大河,最后这条大河流进了大海。这条大河的入海口处有一个村庄——名叫Bytetown。

在Byteland国,有n个伐木的村庄,这些村庄都座落在河边。目前在Bytetown,有一个巨大的伐木场,它处理着全国砍下的所有木料。木料被砍下后,顺着河流而被运到Bytetown的伐木场。Byteland的国王决定,为了减少运输木料的费用,再额外地建造k个伐木场。这k个伐木场将被建在其他村庄里。这些伐木场建造后,木料就不用都被送到Bytetown了,它们可以在运输过程中第一个碰到的新伐木场被处理。显然,如果伐木场座落的那个村子就不用再付运送木料的费用了。它们可以直接被本村的伐木场处理。

注:所有的河流都不会分叉,形成一棵树,根结点是Bytetown。

国王的大臣计算出了每个村子每年要产多少木料,你的任务是决定在哪些村子建设伐木场能获得最小的运费。其中运费的计算方法为:每一吨木料每千米1分钱。

输入输出格式

输入格式:

第一行包括两个数 n(2≤n≤100),k(1≤k≤50,且 k≤n)。n为村庄数,k为要建的伐木场的数目。除了Bytetown外,每个村子依次被命名为1,2,3……n,Bytetown被命名为0。

接下来n行,每行3个整数:

wi——每年i村子产的木料的块数(0≤wi≤10000)

vi——离i村子下游最近的村子(即i村子的父结点)(0≤vi≤n)

di——vi到i的距离(千米)。(1≤di≤10000)

保证每年所有的木料流到bytetown的运费不超过2,000,000,000分

50%的数据中n不超过20。

输出格式:

输出最小花费,单位为分。

输入输出样例

输入样例#1: 复制

4 2

1 0 1

1 1 10

10 2 5

1 2 3

输出样例#1: 复制

4

这道题因为村庄呈树形结构,所以选用树形DP。

自底向上更新答案,dp[x][j][k][0/1] 代表在第x节点处,离x最近的祖先伐木场为j,用了k个伐木场,x节点有/无伐木场的最小花费。先dfs遍历树,开一个stack数组记录每个节点到根节点的距离。之后开始枚举j与k与子树伐木场个数,更新答案。

#include<iostream>
#include<cstdio> using namespace std;
typedef long long LL;
const int MAXN = 105; struct Edge{
int nxt,to,val;
}edge[MAXN*2];
//size为stack中元素个数,stack记录子节点,d记录深度。
int n,K,w[MAXN],size,stack[MAXN],d[MAXN];
int head[MAXN],cnt;
LL dp[MAXN][MAXN][MAXN][2]; inline void add(int bg,int ed,int ww){
edge[++cnt].to=ed;
edge[cnt].val=ww;
edge[cnt].nxt=head[bg];
head[bg]=cnt;
} inline void dfs(int u){
stack[++size]=u;
for(register int i=head[u];i;i=edge[i].nxt){
int v=edge[i].to; //枚举子节点。
d[v]=d[u]+edge[i].val; //更新子节点距离。
dfs(v);
for(int j=1;j<=size;j++) //枚举祖先。
for(int k=K;k>=0;k--){ //枚举u及子树的伐木场个数。
dp[u][stack[j]][k][0]+=dp[v][stack[j]][0][0];
dp[u][stack[j]][k][1]+=dp[v][u][0][0];
//子节点不建。
for(int x=0;x<=k;x++){ //枚举子节点建的伐木场数。
dp[u][stack[j]][k][0]=min(dp[u][stack[j]][k-x][0]
+dp[v][stack[j]][x][0],dp[u][stack[j]][k][0]);
dp[u][stack[j]][k][1]=min(dp[u][stack[j]][k][1]
,dp[u][stack[j]][k-x][1]+dp[v][u][x][0]);
}
}
}
for(register int j=1;j<=size;j++)
for(register int k=0;k<=K;k++){
if(k>=1){
dp[u][stack[j]][k][0]=min(dp[u][stack[j]][k][0]
+w[u]*(d[u]-d[stack[j]]),dp[u][stack[j]][k-1][1]);
}
else
dp[u][stack[j]][k][0]+=w[u]*(d[u]-d[stack[j]]);
}
//将dp[][][][0]与dp[][][][1]合并。
size--;
}
int main(){
scanf("%d%d",&n,&K);
for(register int i=1;i<=n;i++){
int fa,ww;
scanf("%d%d%d",&w[i],&fa,&ww);
add(fa,i,ww);
}
dfs(0);
printf("%lld",dp[0][0][K][0]);
return 0;
}

IOI 2005 River (洛谷 3354)的更多相关文章

  1. 洛谷3354(IOI2005)河流——“承诺”

    题目:https://www.luogu.org/problemnew/show/P3354 虽说是几个月前曾经讲过的题,但没有题解而自己(花了两个多小时)A了好高兴!!! 这是一个很好的套路:“承诺 ...

  2. bzoj 2005 & 洛谷 P1447 [ Noi 2010 ] 能量采集 —— 容斥 / 莫比乌斯反演

    题目:bzoj 2005 https://www.lydsy.com/JudgeOnline/problem.php?id=2005   洛谷 P1447 https://www.luogu.org/ ...

  3. 【洛谷2904/BZOJ1617】[USACO08MAR]跨河River Crossing(动态规划)

    题目:洛谷2904 分析: 裸dp-- dp方程也不难想: \(dp[i]\)表示运\(i\)头牛需要的最短时间,\(sum[i]\)表示一次运\(i\)头牛(往返)所需的时间,则 \[dp[i]=m ...

  4. 洛谷P1052 过河

    P1052 过河 题目描述 在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上. 由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青 ...

  5. 关于三目运算符与if语句的效率与洛谷P2704题解

    题目描述 司令部的将军们打算在N*M的网格地图上部署他们的炮兵部队.一个N*M的地图由N行M列组成,地图的每一格可能是山地(用“H” 表示),也可能是平原(用“P”表示),如下图.在每一格平原地形上最 ...

  6. 【洛谷】P1052 过河【DP+路径压缩】

    P1052 过河 题目描述 在河上有一座独木桥,一只青蛙想沿着独木桥从河的一侧跳到另一侧.在桥上有一些石子,青蛙很讨厌踩在这些石子上.由于桥的长度和青蛙一次跳过的距离都是正整数,我们可以把独木桥上青蛙 ...

  7. 【题解】洛谷P1350 车的放置(矩阵公式推导)

    洛谷P1350:https://www.luogu.org/problemnew/show/P1350 思路 把矩阵分为上下两块N与M 放在N中的有i辆车 则放在M中有k-i辆车 N的长为a   宽为 ...

  8. 【题解】洛谷P2822 [NOIP2016TG ]组合数问题 (二维前缀和+组合数)

    洛谷P2822:https://www.luogu.org/problemnew/show/P2822 思路 由于n和m都多达2000 所以暴力肯定是会WA的 因为整个组合数是不会变的 所以我们想到存 ...

  9. 洛谷 2216 [HAOI2007]理想的正方形

    题目戳这里 一句话题意 给你一个a×b的矩形,求一个n×n的子矩阵,矩阵里面的最大值和最小值之差最小. Solution 这个题目许多大佬都是单调队列,但是我不是很会,只好用了比较傻逼的方法: 首先我 ...

随机推荐

  1. VS2017编译64位CloudCompare

    需求:编译一个支持读写las点云的CC,然后再开发CC插件实现业务功能. 编译环境: 1.Windows 10 2.Visual Studio 2017 Community 3.Qt 5.9.4 开源 ...

  2. 理解 Java 序列化

    一.什么是序列化 序列化是一种对象持久化的手段.类通过实现 java.io.Serializable 接口以启用其序列化功能. 序列化:把对象转换为字节序列的过程. 反序列化:把字节序列恢复为对象的过 ...

  3. Python爬虫实战—— Request对象之header伪装策略

    在header当中,我们经常会添加两个参数--cookie 和 User-Agent,来模拟浏览器登录,以此提高绕过后台服务器反爬策略的可能性. User-Agent获取 User-Agent可通过随 ...

  4. Vue学习笔记【16】——vue-resource 实现 get, post, jsonp请求

    除了 vue-resource 之外,还可以使用 axios 的第三方包实现实现数据的请求 之前的学习中,如何发起数据请求?原生.jQuery,需要操作DOM 常见的数据请求类型? get post ...

  5. 使用用Intellij Idea从Github上获取代码

    1.打开File菜单,选择Setting,在Version Control下找到Github. 2.分别在Login与Password中输入自己在Github注册的用户名和密码,然后点击Test按钮: ...

  6. BZOJ 3159: 决战 解题报告

    BZOJ 3159: 决战 1 sec 512MB 题意: 给你一颗\(n\)个点,初始点权为\(0\)的有跟树,要求支持 Increase x y w 将路径\(x\)到\(y\)所有点点权加上\( ...

  7. 深入理解js——构造函数的继承

    看了阮一峰的网络日志(http://www.ruanyifeng.com/blog/2010/05/object-oriented_javascript_inheritance.html),记录一下构 ...

  8. 高级运维(五):构建memcached服务、LNMP+memcached、使用Tomcat设置Session、Tomcat实现session共享

    一.构建memcached服务 目标: 本案例要求先快速搭建好一台memcached服务器,并对memcached进行简单的添.删.改.查操作: 1> 安装memcached软件,并启动服务d ...

  9. idea 中使用dataBase插件

    最近一段时间重新开始学习Java 使用IntelliJ 发现有个链接数据库插件挺好用的,并且反应速度还挺快的记录下. 点击步骤:View → Tool Windows → data 进行数据库链接 发 ...

  10. codeforces 447E or 446C 线段树 + fib性质或二次剩余性质

    CF446C题意: 给你一个数列\(a_i\),有两种操作:区间求和:\(\sum_{i=l}^{r}(a[i]+=fib[i-l+1])\).\(fib\)是斐波那契数列. 思路 (一) codef ...