HDU 4003 [树][贪心][背包]

时间:2022-04-27 05:37:31
/*
大连热身A题
不要低头,不要放弃,不要气馁,不要慌张
题意:
给一棵树,每条边上有权值。给一个起点,放置n个机器人,要求使得任意一个节点至少被一个机器人经过。
每个机器人经过某条边时的代价为这条边的权值。反复经过需要反复累积。
问最小的代价是什么。 思路:
1.转化为背包问题。考虑给某个子树i个机器人的最小代价,即有i个机器人跑到某棵子树不回来。其中0个代表给某子树一个机器人,该机器人
遍历完该子树所有节点以后又返回该节点的代价。然后相当于每棵子树有几个物品,至少从中选择一个。进行分组背包。
2.为什么以上转化是成立的,即是否有可能把某个机器人指派到某棵子树以后,该机器人又跑到其它子树,最后停留在其它子树,而能使得花
费的代价更优。事实是不可能的,因为如果该节点跑到子树的某个叶子节点又返回母亲节点,代价肯定比一开始就指派给该子树的机器人,让
它先走到那个节点所经过的叶子节点,再回到子树的根花费的代价要大。所以可以将这个问题转化为分组背包。 */
#include<bits/stdc++.h>
#define N 10060
using namespace std;
long long dp[N][][];
int k;
int fa[N];
long long inf=0x3f3f3f3f3f3f3f3f;
struct edge{
int id;
long long w;
edge *next;
};
int ednum;
edge *adj[N];
edge edges[N<<];
inline void addedge(int a,int b,long long w){
edge *tmp=&edges[ednum++];
tmp->id=b;
tmp->w=w;
tmp->next=adj[a];
adj[a]=tmp;
}
void dfs(int pos){
bool ok=;
for(edge *it=adj[pos];it;it=it->next){
if(!fa[it->id]){
ok=;
fa[it->id]=pos;
dfs(it->id);
dp[pos][][]+=dp[it->id][][]+*it->w;
}
}
if(ok){
for(int i=;i<=k;i++)dp[pos][][i]=inf;
for(edge *it=adj[pos];it;it=it->next){
memset(dp[pos][],0x3f,sizeof(dp[pos][]));
if(fa[it->id]==pos){
for(int i=;i<=k;i++){
for(int j=i;j<=k;j++){
if(dp[pos][][j-i]!=inf)
dp[pos][][j]=min(dp[pos][][j-i]-dp[it->id][][]+(i-)*it->w+dp[it->id][][i],dp[pos][][j]);
}
}
for(int i=;i<=k;i++)dp[pos][][i]=min(dp[pos][][i],dp[pos][][i]);
}
}
}
}
int main()
{
int n,s;
while(scanf("%d%d%d",&n,&s,&k)!=EOF){
memset(adj,NULL,sizeof(adj));
ednum=;
for(int i=;i<n;i++){
int a,b;
long long w;
scanf("%d%d%lld",&a,&b,&w);
addedge(a,b,w);
addedge(b,a,w);
}
memset(fa,,sizeof(fa));
fa[s]=s;
memset(dp,,sizeof(dp));
dfs(s);
long long ans=inf;
for(int i=;i<=k;i++)ans=min(ans,dp[s][][i]);
printf("%lld\n",ans);
}
}