Treats for the Cows 区间DP POJ 3186
题目来源:http://poj.org/problem?id=3186
(http://www.fjutacm.com/Problem.jsp?pid=1389)
/**
题目意思:
约翰经常给产奶量高的奶牛发特殊津贴,于是很快奶牛们拥有了大笔不知该怎么花的钱.
为此,约翰购置了N(1≤N≤2000)份美味的零食来卖给奶牛们.每天约翰售出一份零食.
当然约翰希望这些零食全部售出后能得到最大的收益.这些零食有以下这些有趣的特性:
零食按照1..N编号,它们被排成一列放在一个很长的盒子里.盒子的两端都有开口,约翰每
天可以从盒子的任一端取出最外面的一个.
与美酒与好吃的奶酪相似,这些零食储存得越久就越好吃.当然,这样约翰就可以把它们卖出更高的价钱.
每份零食的初始价值不一定相同.约翰进货时,第i份零食的初始价值为Vi(1≤Vi≤1000).
第i份零食如果在被买进后的第a天出售,则它的售价是vi×a.
Vi的是从盒子顶端往下的第i份零食的初始价值.约翰告诉了你所有零食的初始价值,
并希望你能帮他计算一下,在这些零食全被卖出后,他最多能得到多少钱.
**/
思路:这是一题区间动态规划,但是又不是那么复杂的区间动态规划;
这题要先找对状态,不然很难进行下去,也许别的小伙伴有别的状态,但是我找的状态是dp[i][j]表示i~j范围内取出倒数j-i+1个物品的最大值,比如dp[i][i]就是第取出倒数第一个为a[i]可以获得的值,dp[i-1][i]同理,表示倒数第一、二个为a[i-1],a[i]的最大值。
然后找对状态了要找转移方程:对于这题先要搞清楚我们是逆推,从区间小推向区间大,也就是用小的求大的,所以dp[i][j]是由比他小的部分加上一些东西组成的,此时我们可以发现,dp[i][j]这个区间我们只有两种取法(题目给的是一个双端队列一样的东西),也就是我们只能取a[j]或者a[i]这两个数,而且这两个数一定是在倒数j-i次取的,因为之前只有j-i- 1(可能是a[i]或a[j]) 个数,而且都是倒数着的,这样我们就可以发现,dp[i][j]=Max(dp[i+1][j]+a[i]*(n-(j-i)), dp[i][j-1]+a[j]*(n-(j-i)));看下面这个图你就能理解他是怎么步步紧逼的合并状态的:

斜着的颜色分层的部分就表示i、j距离不同时由某些状态过来的最大值,余下的看看代码吧!
#include<stdio.h>
#include<string.h>
const int N=;
int dp[N][N], a[N];
int Max(int a, int b){ return a>b?a:b; }
int main( ){
int n;
while(scanf("%d", &n)!=EOF){
memset(dp, , sizeof(dp));
for(register int i=; i<=n; ++i)
scanf("%d", a+i), dp[i][i]=a[i]*n;///设置为当第i个数为最后一个被取的数的情况;
for(register int l=; l<=n; ++l){///先跑i、j距离为1的区间,然后顺延下去才可以保证不会出错
for(register int i=; i+l<=n; ++i){
int j=i+l;
dp[i][j]=Max(dp[i+][j]+a[i]*(n-l), dp[i][j-]+a[j]*(n-l));
}
}
printf("%d\n", dp[][n]);
}
}
Treats for the Cows 区间DP POJ 3186的更多相关文章
- POJ3186:Treats for the Cows(区间DP)
Description FJ has purchased N (1 <= N <= 2000) yummy treats for the cows who get money for gi ...
- POJ3086 Treats for the Cows(区间DP)
题目链接 Treats for the Cows 直接区间DP就好了,用记忆化搜索是很方便的. #include <cstdio> #include <cstring> #i ...
- O - Treats for the Cows 区间DP
FJ has purchased N (1 <= N <= 2000) yummy treats for the cows who get money for giving vast am ...
- POJ 3186Treats for the Cows(区间DP)
题目链接:http://poj.org/problem?id=3186 题目大意:给出的一系列的数字,可以看成一个双向队列,每次只能从队首或者队尾出队,第n个出队就拿这个数乘以n,最后将和加起来,求最 ...
- poj3186 Treats for the Cows(区间)
题目链接:http://poj.org/problem?id=3186 题意:第一个数是N,接下来N个数,每次只能从队列的首或者尾取出元素. ans=每次取出的值*出列的序号.求ans的最大值. 样例 ...
- 区间DP POJ 1141 Brackets Sequence
Brackets Sequence Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 29520 Accepted: 840 ...
- 【BZOJ】1652: [Usaco2006 Feb]Treats for the Cows(dp)
http://www.lydsy.com/JudgeOnline/problem.php?id=1652 dp.. 我们按间隔的时间分状态k,分别为1-n天 那么每对间隔为k的i和j.而我们假设i或者 ...
- POJ 3186 Treats for the Cows ——(DP)
第一眼感觉是贪心,,果断WA.然后又设计了一个两个方向的dp方法,虽然觉得有点不对,但是过了样例,交了一发,还是WA,不知道为什么不对= =,感觉是dp的挺有道理的,,代码如下(WA的): #incl ...
- POJ 3186Treats for the Cows (区间DP)
详见代码 #include <stdio.h> #include <algorithm> #include <string.h> using namespace s ...
随机推荐
- JVM 规范
http://files.cnblogs.com/files/dragonsuc/jls8.pdf 或者官网:http://files.cnblogs.com/files/dragonsuc/jls8 ...
- HDU2883_kebab
很好的题目. 有不多于200个任务,每个任务要在si到ei这个时间段内完成,每个任务的任务量是ti*ni,只有一台机器,且其单位时间内可完成的任务量为m. 现在问你,能否使所有的任务全部在规定的时间段 ...
- matlab gradient 和 prctile
介绍两个matlab小函数: 1.gradient 借用别人的例子:例:>> x=[6,9,3,4,0;5,4,1,2,5;6,7,7,8,0;7,8,9,10,0]x = 6 ...
- Python实现类似JavaScript 的Json对象
Python实现类似JavaScript 的Json对象 用过js的都知道 js中json也是一个对象,所以可以直接通过class.attr 取值,当attr不存在时也不会报错,那么Python可不可 ...
- php获取星期几周几
PHP星期几获取代码: date("l"); //data就可以获取英文的星期比如Sundaydate("w"); //这个可以获取数字星期比如123,注意0是 ...
- [spark]-Spark2.x集群搭建与参数详解
在前面的Spark发展历程和基本概念中介绍了Spark的一些基本概念,熟悉了这些基本概念对于集群的搭建是很有必要的.我们可以了解到每个参数配置的作用是什么.这里将详细介绍Spark集群搭建以及xml参 ...
- vue2.0实战记录
1. 初始化项目vue init webpack caseone cd caseonecnpm installcnpm install less less-loader -Dcnpm install ...
- Python中的变量引用对象需注意的几点
Python中的变量引用对象需注意的几点 分类:Python (55) (0) 普通引用: Python中,变量的作用仅仅是一个标识,只有赋值后才被创建,它可以引用任何类型的对象,而且在引用之前必须 ...
- 线程中wait/notify/notifyAll的用法
前言 多线程时,最关注的就是线程同步,线程间的同步一般用锁来实现,常见的锁就是synchronized和lock.用了synchronized,就不得不提到wait/notify/notifyAll. ...
- 洛谷 P3382 【模板】三分法
https://www.luogu.org/problem/show?pid=3382 题目描述 如题,给出一个N次函数,保证在范围[l,r]内存在一点x,使得[l,x]上单调增,[x,r]上单调减. ...