Vijos 1523 贪吃的九头龙 【树形DP】
贪吃的九头龙
背景
安徽省芜湖市第二十七中学测试题
NOI 2002 贪吃的九头龙(dragon)
Description:Official
Data:Official
Program:Converted by JackDavid127描述
传说中的九头龙是一种特别贪吃的动物。虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落。
有一天,有M个脑袋的九头龙看到一棵长有N个果子的果树,喜出望外,恨不得一口把它全部吃掉。可是必须照顾到每个头,因此它需要把N个果子分成M组,每组至少有一个果子,让每个头吃一组。
这M个脑袋中有一个最大,称为“大头”,是众头之首,它要吃掉恰好K个果子,而且K个果子中理所当然地应该包括唯一的一个最大的果子。果子由N-1根树枝连接起来,由于果树是一个整体,因此可以从任意一个果子出发沿着树枝“走到”任何一个其他的果子。
对于每段树枝,如果它所连接的两个果子需要由不同的头来吃掉,那么两个头会共同把树枝弄断而把果子分开;如果这两个果子是由同一个头来吃掉,那么这个头会懒得把它弄断而直接把果子连同树枝一起吃掉。当然,吃树枝并不是很舒服的,因此每段树枝都有一个吃下去的“难受值”,而九头龙的难受值就是所有头吃掉的树枝的“难受值”之和。
九头龙希望它的“难受值”尽量小,你能帮它算算吗?
格式
输入格式
输入的第1行包含三个整数N(1<=N<=300),M(2<=M<=N),K(1<=K<=N)。N个果子依次编号1,2,...,N,且最大的果子的编号总是1。第2行到第N行描述了果树的形态,每行包含三个整数a(1<=a<=N),b(1<=b<=N),c(0<=c<=105),表示存在一段难受值为c的树枝连接果子a和果子b。
输出格式
输出仅有一行,包含一个整数,表示在满足“大头”的要求的前提下,九头龙的难受值的最小值。如果无法满足要求,输出-1。
样例1
样例输入1
8 2 4
1 2 20
1 3 4
1 4 13
2 5 10
2 6 12
3 7 15
3 8 5
样例输出1
4
提示
树形动态规划
题目链接:
题目大意:
N个果子的一颗树,树的边权都有一个值。九头龙有M个脑袋,1号脑袋(大头)必须吃1号果子,且1号脑袋必须吃恰好K个果子。
其他的脑袋(小头)至少吃一个果子。如果一个树枝的两段都被同一个头吃掉,那么代价就加上这条边的值。问满足题意的最小代价。
题目思路:
【树形DP】
首先通过观察我们发现这题的M只存在两种情况,M=2和M>2
如果M>2,那么通过交换必然有一种安排使得每个头都有至少一个果子吃且小头不会吃相邻的果子。
所以只需要考虑一个果子是否被大头吃,共两种状态。如果M=2,则必须算上小头吃的相邻果子的代价。
因为是一棵多叉树,转二叉树后为左儿子右兄弟,所以需要记录的是父亲的状态(二叉形态下的右兄弟需要知道原先多叉形态下父亲的状态)
转移的时候f[i][j][k]表示节点i,以i为根的子树(含i)被大头吃掉j个果子,父亲(多叉下)状态为k的最小代价。
k=0表示该节点的父亲(多叉下)被小头吃,k=1表示该节点的父亲(多叉下)被大头吃的代价。
那么只需要枚举当前节点被大头还是小头吃即可。
/****************************************************
Author : Coolxxx
Copyright 2017 by Coolxxx. All rights reserved.
BLOG : http://blog.csdn.net/u010568270
****************************************************/
#include<bits/stdc++.h>
#pragma comment(linker,"/STACK:1024000000,1024000000")
#define abs(a) ((a)>0?(a):(-(a)))
#define lowbit(a) (a&(-a))
#define sqr(a) ((a)*(a))
#define mem(a,b) memset(a,b,sizeof(a))
#define eps (1e-8)
#define J 10000
#define mod 1000000007
#define MAX 0x7f7f7f7f
#define PI 3.14159265358979323
#define N 304
using namespace std;
typedef long long LL;
double anss;
LL aans;
int cas,cass;
int n,m,lll,ans;
int l[N],r[N],q[N],s[N],fa[N];
int ma[N][N];
int f[N][N][];
void dfs(int now,int ff)
{
if(!now)return;
fa[now]=ff;
if(l[now])dfs(l[now],now);
if(r[now])dfs(r[now],now);
s[now]=s[l[now]]+s[r[now]]+;
q[++lll]=now;
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("1.txt","r",stdin);
// freopen("2.txt","w",stdout);
#endif
int i,j,k;
int x,y,z;
// for(scanf("%d",&cass);cass;cass--)
// for(scanf("%d",&cas),cass=1;cass<=cas;cass++)
// while(~scanf("%s",s))
while(~scanf("%d",&n))
{
mem(l,);mem(r,);mem(f,);
scanf("%d%d",&m,&cas);
cass=(m==);
for(i=;i<n;i++)
{
scanf("%d%d%d",&x,&y,&z);
if(x>y)swap(x,y);
if(!l[x])
{
l[x]=y;
ma[x][y]=ma[y][x]=z;
}
else
{
r[y]=l[x];
l[x]=y;
ma[x][y]=ma[y][x]=z;
ma[y][r[y]]=ma[r[y]][y]=ma[x][r[y]];
}
}
if(m+cas>n){puts("-1");continue;}
lll=;
dfs(,);
f[][][]=f[][][]=;
for(x=;x<=lll;x++)
{
i=q[x];
for(j=;j<=min(cas,s[i]);j++)
{
for(y=;y<=j;y++)
{
for(k=;k<=;k++)
{
f[i][j][k]=min(f[i][j][k],f[l[i]][y][]+f[r[i]][j-y][k]+ma[i][fa[i]]*(k==)*cass);
if(y>)f[i][j][k]=min(f[i][j][k],f[l[i]][y-][]+f[r[i]][j-y][k]+ma[i][fa[i]]*(k==));
}
}
}
}
printf("%d\n",f[l[]][cas-][]);
}
return ;
}
/*
//
//
*/
Vijos 1523 贪吃的九头龙 【树形DP】的更多相关文章
- Vijos1523 NOI2002 贪吃的九头龙 树形dp
思路不算很难,但细节处理很麻烦 前面建图.多叉转二叉,以及确定dp处理序列的过程都是套路,dp的状态转移过程以注释的形式阐述 #include <cstdio> #include < ...
- vijos p1523 贪吃的九头龙 思考思考再思考,就荒废了4小时
树形DP要有自己的风格,转二叉树是基础,考虑边界最头疼. #include<cstdio> #include<cstring> #include<algorithm> ...
- [NOI2002]贪吃的九头龙(树形dp)
[NOI2002]贪吃的九头龙 题目背景 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是 说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的 ...
- Vijos1523贪吃的九头龙【树形DP】
贪吃的九头龙 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头 ...
- 贪吃的九头龙(tyvj P1523)
T2 .tyvj P1523贪吃的九头龙 描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于 ...
- [codevs1746][NOI2002]贪吃的九头龙
[codevs1746][NOI2002]贪吃的九头龙 试题描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫"九头龙",但这只是说它出生的时候有九个头,而在成长的过程中,它有时 ...
- vojis1523 NOI2002 贪吃的九头龙
描述 传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落. 有一天, ...
- codevs1746 贪吃的九头龙
[问题描述]传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落.有一 ...
- codevs贪吃的九头龙
传说中的九头龙是一种特别贪吃的动物.虽然名字叫“九头龙”,但这只是说它出生的时候有九个头,而在成长的过程中,它有时会长出很多的新头,头的总数会远大于九,当然也会有旧头因衰老而自己脱落.有一天,有M 个 ...
随机推荐
- 循环中i++和++i哪个好
推荐使用++i,因为不需要返回临时对象,执行效率更高.
- centos7配置静态IP步骤
centos7按照初始安装时候的developer类型一路装好,在vmware里已经设置为bridge模式,按理说是会自动按照DHCP联网成功的,结果却发现连网卡都没有激活,这里记录下. 1:我要把L ...
- Centos6.5下 执行“ll”提示“-bash: ll: command not found”
ll 是 ls -l的别名,之所所以 ll出现错误是因为没有定义别名. 如果要实现ll 命令,可以做如下操作: 编辑 ~./bashrc 添加 ls -l 的别名为 ll即可 [root@Centos ...
- 大前端之HTML5\CSS3
- Python之爬虫-京东商品
Python之爬虫-京东商品 #!/usr/bin/env python # coding: utf-8 from selenium import webdriver from selenium.we ...
- 一个ajax实例
一个ajax实例 html <!DOCTYPE html> <html lang="zh-cn"> <head> <meta ch ...
- BNUOJ 26229 Red/Blue Spanning Tree
Red/Blue Spanning Tree Time Limit: 2000ms Memory Limit: 131072KB This problem will be judged on HDU. ...
- golang函数指针的效果
package main import ( "fmt" ) func fun1(key string) { fmt.Printf("fun11 key=%s\n" ...
- [luoguP1086] 花生采摘(模拟)
传送门 模拟... 代码 #include <cstdio> #include <iostream> #include <algorithm> #define ab ...
- 【最大流】Escape
https://www.bnuoj.com/v3/contest_show.php?cid=9149#problem/F [题意] 给定n个人和m个星球,每个人可以匹配某些星球,每个星球有一定的容量限 ...