1812: [Ioi2005]riv

1812: [Ioi2005]riv

Time Limit: 10 Sec Memory Limit: 64 MB

Submit: 635 Solved: 388

[Submit][Status][Discuss]

Description

几乎整个Byteland王国都被森林和河流所覆盖。小点的河汇聚到一起,形成了稍大点的河。就这样,所有的河水都汇聚并流进了一条大河,最后这条大河流进了大海。这条大河的入海口处有一个村庄——名叫Bytetown 在Byteland国,有n个伐木的村庄,这些村庄都座落在河边。目前在Bytetown,有一个巨大的伐木场,它处理着全国砍下的所有木料。木料被砍下后,顺着河流而被运到Bytetown的伐木场。Byteland的国王决定,为了减少运输木料的费用,再额外地建造k个伐木场。这k个伐木场将被建在其他村庄里。这些伐木场建造后,木料就不用都被送到Bytetown了,它们可以在 运输过程中第一个碰到的新伐木场被处理。显然,如果伐木场座落的那个村子就不用再付运送木料的费用了。它们可以直接被本村的伐木场处理。 注意:所有的河流都不会分叉,也就是说,每一个村子,顺流而下都只有一条路——到bytetown。 国王的大臣计算出了每个村子每年要产多少木料,你的任务是决定在哪些村子建设伐木场能获得最小的运费。其中运费的计算方法为:每一块木料每千米1分钱。 编一个程序: 1.从文件读入村子的个数,另外要建设的伐木场的数目,每年每个村子产的木料的块数以及河流的描述。 2.计算最小的运费并输出。

Input

第一行 包括两个数 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村子下游最近的村子(或bytetown)(0<=vi<=n) di——vi到i的距离(km)。(1<=di<=10000) 保证每年所有的木料流到bytetown的运费不超过2000,000,000分 50%的数据中n不超过20。

Output

输出最小花费,精确到分。

Sample Input

4 2

1 0 1

1 1 10

10 2 5

1 2 3

Sample Output

4

HINT

Source

[Submit][Status][Discuss]



万恶的树形dp,写了一下午QAQ

  • 如果有一道题是wzx没有做过的,唯一可能是他并不想做

嘤嘤嘤

二位状态不太合理,要建三维

\(f[i][j][k]\)表示在\(i\)的子树中一共建了k个伐木场,其中没有到伐木场的木头需要运到\(j\)级祖先

主要是难转移啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊啊


#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define max(a,b) ((a)>(b) ? (a): (b))
#define min(a,b) ((a)<(b) ? (a): (b))
#define LL long long
#define M 999999999999
#define RI register int
using namespace std;

int i,m,n,j,k,a[101],d[101],ver[101],edge[101],head[101],nex[101],w[101],v,g,cnt,s[101];
LL f[101][101][101];
inline void add(int x,int y,int z)
{
    cnt+=1; ver[cnt]=y; nex[cnt]=head[x]; head[x]=cnt; edge[cnt]=z;
}

void dfs(int now,int de)
{
    s[now]=1;
    for(RI i=head[now];i;i=nex[i])
    {
        int t=ver[i];
        d[de+1]=edge[i]+d[de];
        dfs(t,de+1);
        s[now]+=s[t];
        for(RI j=0;j<=de;j++)
            for(RI l=s[now];l>=0;l--)
            {
                LL x=M;
                for(RI z=l;z>=0;z--) x=(LL)min(x,min(f[now][j][l-z]+f[t][0][z],f[now][j][l-z]+f[t][j+1][z]));
                f[now][j][l]=x;
            }
    }

    for(RI i=1;i<=de;i++)
        for(RI j=0;j<=s[now]-1;j++) f[now][i][j]+=(LL)(d[de]-d[de-i])*w[now];
    for(RI i=k+1;i;i--) f[now][0][i]=(LL)f[now][0][i-1];
    f[now][0][0]=M;
}

int main()
{
    scanf("%d%d",&n,&k);
    for(RI i=1;i<=n;i++)
    {
        scanf("%d%d%d",&w[i],&v,&g);
        add(v,i,g);
    }
    dfs(0,0);
    printf("%lld",f[0][0][k+1]);
}

原文地址:https://www.cnblogs.com/ZUTTER/p/9794999.html

时间: 2024-08-29 20:30:56

1812: [Ioi2005]riv的相关文章

BZOJ1812: [Ioi2005]riv

1812: [Ioi2005]riv Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 523  Solved: 309[Submit][Status][Discuss] Description 几 乎整个Byteland王国都被森林和河流所覆盖.小点的河汇聚到一起,形成了稍大点的河.就这样,所有的河水都汇聚并流进了一条大河,最后这条大河流进 了大海.这条大河的入海口处有一个村庄——名叫Bytetown 在Byteland国,有n个伐木的村庄,这些村庄

[IOI2005]Riv 河流

题目描述 几乎整个Byteland 王国都被森林和河流所覆盖.小点的河汇聚到一起,形成了稍大点的河.就这样,所有的河水都汇聚并流进了一条大河,最后这条大河流进了大海.这条大河的入海口处有一个村庄--Bytetown.在Byteland国,有n个伐木的村庄,这些村庄都座落在河边.目前在Bytetown,有一个巨大的伐木场,它处理着全国砍下的所有木料.木料被砍下后,顺着河流而被运到Bytetown的伐木场.Byteland 的国王决定,为了减少运输木料的费用,再额外地建造k个伐木场.这k个伐木场将被

bzoj1812 [IOI2005]riv河流

题目链接 problem 给出一棵树,每个点有点权,每条边有边权.0号点为根,每个点的代价是这个点的点权\(\times\)该点到根路径上的边权和. 现在可以选择最多K个点.使得每个点的代价变为:这个点的点权\(\times\)改点到最近的被选中的一个祖先的边权和. 问所有点的代价和最小为多少. solution 用\(g[i][j]\)表示以i为根的子树,强制选i,最大的贡献(这里的贡献是指比什么也不选所减少的代价.) 最终答案肯定就是初始代价-g[0][k] 考虑怎么维护出\(g\).用\(

YCB 的暑期计划

前言 YCB现在很弱(TAT) 暑假有一个月,赶快狂补一下. 大概的计划如下: 首先前期会以数据结构为主,毕竟代码能力太弱,涉及内容:线段树分治.二进制分组.KD-Tree. 等数据结构做到没有智商的时候加入一波数论,内容为 杜教筛.min_25筛. 然后中途小清新一下,做一些 组合博弈与构造题. 接着继续练代码能力,顺便学一些神奇的暴力:启发式合并.dsu on tree . 然后图论也忘的差不多了,就回过头去学点新东西,大概会有spfa判负环.0/1分数规划.差分约束. 估计这个时候也没有什

带权二分

带权二分 一种二分答案的套路,又叫做DP凸优化,wqs二分. 用来解决一类题目,要求某个要求出现K次,并且,可以很显然的发现,在改变相应权值的时候,对应出现的次数具有单调性.而且很显然,这种题一般满足一定的要求.而且一般权值为整数二分就可以,但是有的题需要实数二分...而且,边界条件通常很麻烦,调起来想摔电脑. 例题时间: BZOJ2654: tree 题目大意:给你一个图,里面有白色边和黑色边,问恰好有k条边白色边的最小生成树 直接贪心法肯定是错误的,因此,我们考虑带权二分. 给定一个选择白色

IOI 2005/bzoj 1812:riv 河流

Description 几乎整个Byteland王国都被森林和河流所覆盖.小点的河汇聚到一起,形成了稍大点的河.就这样,所有的河水都汇聚并流进了一条大河,最后这条大河流进了大海.这条大河的入海口处有一个村庄--名叫Bytetown 在Byteland国,有n个伐木的村庄,这些村庄都座落在河边.目前在Bytetown,有一个巨大的伐木场,它处理着全国砍下的所有木料.木料被砍下后,顺着河流而被运到Bytetown的伐木场.Byteland的国王决定,为了减少运输木料的费用,再额外地建造k个伐木场.这

csu 1812: 三角形和矩形 凸包

传送门:csu 1812: 三角形和矩形 思路:首先,求出三角形的在矩形区域的顶点,矩形在三角形区域的顶点.然后求出所有的交点.这些点构成一个凸包,求凸包面积就OK了. /************************************************************** Problem: User: youmi Language: C++ Result: Accepted Time: Memory: ***********************************

spoj 1812 LCS2 - Longest Common Substring II (后缀自动机)

spoj 1812 LCS2 - Longest Common Substring II 题意: 给出最多n个字符串A[1], ..., A[n], 求这n个字符串的最长公共子串. 限制: 1 <= n <= 10 |A[i]| <= 1e5 思路: 和spoj 1811 LCS差不多的做法 把其中一个A建后缀自动机 考虑一个状态s, 如果A之外的其他串对它的匹配长度分别是a[1], a[2], ..., a[n - 1], 那么min(a[1], a[2], ..., a[n - 1]

spoj 1812

1812. Longest Common Substring II Problem code: LCS2 A string is finite sequence of characters over a non-empty finite set Σ. In this problem, Σ is the set of lowercase letters. Substring, also called factor, is a consecutive sequence of characters o