HDU 4003 Find Metal Mineral
这个题是POJ1849的加强版。
先说一个很重要的结论,下面两种方法都是从这个结论出发的。
一个人从起点遍历一颗树,如果最终要回到起点,走过的最小权值就是整棵树的权值的2倍。
而且K个人的情况也是如此,大不了只有一个人走,其他K-1个人待着不动就行了。
而题目中说了这些人不比回到原点,所以就想办法考虑哪些多走的路程,最后用整棵树权值2倍减去就好了。
一、
多数人的做法是分组背包,推荐这篇博客。
里面的状态的定义并不复杂,和网上那些千篇一律的做法比,思路很新颖。
f(i, j)表示i为根的子树有j个机器人出发,遍历这棵子树可以少走多少路。
最终答案为sum - f(S, K)
#include <iostream>
#include <cstdio>
#include <cstring>
#include <vector>
#include <algorithm>
using namespace std; const int maxn = + ;
const int maxk = ; int n, s, k; vector<int> G[maxn], C[maxn]; int f[maxn][maxk]; void DP(int u, int fa)
{
for(int i = ; i < G[u].size(); i++)
{
int v = G[u][i], w = C[u][i];
if(v == fa) continue;
DP(v, u);
for(int i = k; i >= ; i--)
for(int j = ; j <= i; j++)
f[u][i] = max(f[u][i], f[u][i-j] + f[v][j] + ( - j) * w);
}
} int main()
{
while(scanf("%d%d%d", &n, &s, &k) == )
{
for(int i = ; i <= n; i++) { G[i].clear(); C[i].clear(); } int sum = ;
for(int i = ; i < n; i++)
{
int u, v, w; scanf("%d%d%d", &u, &v, &w);
sum += w;
G[u].push_back(v); C[u].push_back(w);
G[v].push_back(u); C[v].push_back(w);
}
sum <<= ; memset(f, , sizeof(f));
DP(s, );
printf("%d\n", sum - f[s][k]);
} return ;
}
代码君
二、
还有一种更为巧妙的做法,就是每次从起点出发找到一条最长的路径,记录下路径的长度,然后把这条路径上边的权值变为相反数。注意,已经变为负权的边就不再变回去了。
这样找K次最长的路径,然后加起来就是可以最多少走多少路。
为什么要把权值变成负的,因为你第二次第三次再走这条路的时候,就表示少走了负的权值,也就是多走了。如果出现所有的路都是负权的情况,那么树中最长路的路径就是0,也就是机器人原地不动。
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <vector>
using namespace std; const int maxn = + ; struct Edge
{
int u, v, w;
Edge(int u, int v, int w):u(u), v(v), w(w) {}
}; vector<Edge> edges;
vector<int> G[maxn]; void AddEdge(int u, int v, int w)
{
edges.push_back(Edge(u, v, w));
edges.push_back(Edge(v, u, w));
int m = edges.size();
G[u].push_back(m - );
G[v].push_back(m - );
} int n, s, k; int len, id;
int pre[maxn]; void dfs(int u, int fa, int d)
{
if(d > len) { len = d; id = u; }
for(int i = ; i < G[u].size(); i++)
{
Edge& e = edges[G[u][i]];
int v = e.v;
if(v == fa) continue;
pre[v] = G[u][i];
dfs(v, u, d + e.w);
}
} int main()
{
while(scanf("%d%d%d", &n, &s, &k) == )
{
edges.clear();
int tot = ;
for(int i = ; i <= n; i++) G[i].clear();
for(int i = ; i < n; i++)
{
int u, v, w; scanf("%d%d%d", &u, &v, &w);
tot += w;
AddEdge(u, v, w);
}
tot <<= ; int ans = ;
for(int i = ; i < k; i++)
{
len = , id = s;
pre[s] = -;
dfs(s, , );
if(id == s) continue;
ans += len;
for(int u = id; u != s; u = edges[pre[u]].u)
{
int& w = edges[pre[u]].w;
if(w > ) w = -w;
}
} printf("%d\n", tot - ans);
} return ;
}
代码君
HDU 4003 Find Metal Mineral的更多相关文章
- hdu 4003 Find Metal Mineral 树形DP
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4003 Humans have discovered a kind of new metal miner ...
- hdu 4003 Find Metal Mineral 树形dp ,*****
Find Metal Mineral Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65768/65768 K (Java/Other ...
- HDU 4003 Find Metal Mineral(分组背包+树形DP)
题目链接 很棒的一个树形DP.学的太渣了. #include <cstdio> #include <string> #include <cstring> #incl ...
- HDU 4003 Find Metal Mineral (树形DP,经典)
题意:给定一棵树图,n个节点,有边权,要派k<11个机器人从节点s出发,遍历所有的点,每当1只机器人经过1条边时就会花费该边的边权,边是可重复走的.问遍历完所有点的最小花费? 思路: 非常经典, ...
- HDOJ 4003 Find Metal Mineral
题意: 一棵有权树,从根结点中放入 K 个机器人.求用这 K 个机器人遍历全部的结点最少的权值和. 思路: 1. dp[u][i] 表示给以 u 为根节点的子树放 i 个机器人,遍历其子树所须要的最小 ...
- xtu summer individual 6 E - Find Metal Mineral
Find Metal Mineral Time Limit: 1000ms Memory Limit: 65768KB This problem will be judged on HDU. Orig ...
- HDU4003Find Metal Mineral[树形DP 分组背包]
Find Metal Mineral Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65768/65768 K (Java/Other ...
- HDU 4003 (树形DP+背包)
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=4003 题目大意:有K个机器人,走完树上的全部路径,每条路径有个消费.对于一个点,机器人可以出去再回来 ...
- HDU4003 Find Metal Mineral 树形DP
Find Metal Mineral Problem Description Humans have discovered a kind of new metal mineral on Mars wh ...
随机推荐
- SpringMVC-高级参数绑定
绑定数组 需求 在商品列表页面选中多个商品,然后删除. 需求分析 此功能要求商品列表页面中的每个商品前有一个checkbook,选中多个商品后点击删除按钮把商品id传递给Controller,根据商品 ...
- Springboot日志配置探索(主要看logback)(一)
这篇博客是springboot日志配置探索的第一篇,主要讲默认配置下springboot的logback日志框架的配置(即直接使用是怎样的) 首先,是一个SpringBoot的有关日志的说明文档:ht ...
- codeforces1025
hackforces + fstforces A 很明显当有一个字母出现次数>1时即合法 $n = 1$的情况需要特判 #include<cstdio> #include<ve ...
- jQuery源码分析系列(转载来源Aaron.)
声明:非本文原创文章,转载来源原文链接Aaron. 版本截止到2013.8.24 jQuery官方发布最新的的2.0.3为准 附上每一章的源码注释分析 :https://github.com/JsAa ...
- JavaScript 30 - 3 学习笔记
今天学习的是JavaScript 30-3 ---css Variables 实现的效果如下图所示. 废话不多,我们直接来看代码. html: <h1>大家好,这个一个<span c ...
- jquery中ajax请求后台数据成功后既不执行success也不执行error解决方法
jquery中ajax请求后台数据成功后既不执行success也不执行error,此外系统报错:Uncaught SyntaxError: Unexpected identifier at Objec ...
- arcgis jsapi接口入门系列(2):图层基础操作
//图层相关demo layerFun: function () { //获取地图的所有图层(不包括的图层类型:底图图层(basemaps)) let layers = this.map.layers ...
- “Debug Assertion” Runtime Error on VS2008 VS2010 winhand.cpp
I'm writing a C++ MFC program on VS2008 and I'm getting this "Debug Assertion Error" when ...
- 绘制surfaceView 基础类
public class SurfaceViewTempalte extends SurfaceView implements Callback, Runnable { private Surface ...
- Memcache笔记01-Memcache的安装
memcached是什么? Memcached是一个高性能的分布式的内存对象缓存系统,目前全世界不少人使用这个缓存项目来构建自己大负载的网站,来分担数据库的压力,通过在内存里维护一个统一的巨大的has ...