博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
HDU1421:搬寝室(线性dp)
阅读量:5213 次
发布时间:2019-06-14

本文共 1139 字,大约阅读时间需要 3 分钟。

题目:

又是一道,没有思想的题,看了题解,我发现我的dp题几乎都看了题解,我总是想不好状态转移方程,汗颜,以后怎么比赛啊。

先排序,然后说一个数学问题。

首先,要怎么搬呢?即每一对要怎么取?如果有abcd四个数,且a<b<c<d,应该是取ab,cd好呢还是ac,bd好?抑或是bc,ad好呢?答案是第一种,因为:

(a-b)^2+(c-d)^2 < (a-c)^2+(b-d)^2(a-b)^2+(c-d)^2 < (a-d)^2+(b-c)^2

即每对物品都应是重量最为接近的物品,也就是说对n件物品排序后,每对物品都应该是连续的

定义数组w[i]为搬第i对物品所消耗的疲劳值;数组dp[n][k]来表示在n件物品中搬k对的最佳状态,而达到这一状态的决策可能为:

  1. 第n件物品不搬,即在前n - 1件物品中搬k对,那么疲劳值仍为dp[n - 1][k];
  2. 第n件物品要搬,那么根据上面所证,第n - 1件物品也要同时搬,即在前n - 2件物品中搬k - 1对物品,再搬最后一对物品,那么疲劳值为dp[n - 2][k - 1] + w[n - 1]。

为使疲劳值最小,因此最佳策略为取两种决策中的最小值,即应使:

dp[n][k] = min(dp[n - 1][k], dp[n - 2][k - 1] + w[n - 1])

 

应该注意的是要考虑边界问题,dp[i][j]中:

  1. 当2 * j > i时,即要搬的数量超过了物品总量,这是不可能发生的,因此此时令dp[i][j]为无穷大;
  2. 当j == 0时,即在一对物品都没搬时,所需疲劳值应该是0,此时令dp[i][j] = 0。
#include 
#include
#include
#include
#include
#include
#define inf 0x3f3f3f3ftypedef long long ll;using namespace std;int n,k,w[2010],dp[2010][1010];int main(){ while(scanf("%d%d",&n,&k)!=EOF) { for(int i=1;i<=n;i++) { scanf("%d",&w[i]); } sort(w+1,w+1+n); for(int i=1;i

 

转载于:https://www.cnblogs.com/zhangmingcheng/p/4372900.html

你可能感兴趣的文章
LeetCode 16. 3Sum Closest
查看>>
将文本文件中的\n字符串变成换行符
查看>>
如何在博客园设置自己的头像
查看>>
NIO选择器学习笔记
查看>>
Nginx配置upstream实现负载均衡1
查看>>
“ipconfig不是内部命令或外部命令”解决方法
查看>>
linux cron定时任务初级使用教程
查看>>
(C#控件)MessageBox
查看>>
Excel:写入Excel-单纯写入
查看>>
Tomcat详细用法学习(五)
查看>>
2017 icpc亚洲区域赛沈阳站
查看>>
UI基础--封装cell滑动时的动画
查看>>
2017.9.1 Java中的程序方法
查看>>
Django 框架 基础
查看>>
HDU3306 Another kind of Fibonacci 矩阵
查看>>
CSS笔记-文本缩略显示
查看>>
S7-200PLC间的PPI通信
查看>>
第三章家庭作业3.65
查看>>
javascript有哪些优秀的库,把你喜欢的都说出来吧
查看>>
Web后端 JAVA学习之路
查看>>