第十一节 动态规划 - 3

1|0区间 DP

2|0A. 能量项链

题目描述

在 Mars 星球上,每个 Mars 人都随身佩带着一串能量项链。在项链上有 N 颗能量珠。能量珠是一颗有头标记与尾标记的珠子,这些标记对应着某个正整数。并且,对于相邻的两颗珠子,前一颗珠子的尾标记一定等于后一颗珠子的头标记。因为只有这样,通过吸盘(吸盘是 Mars 人吸收能量的一种器官)的作用,这两颗珠子才能聚合成一颗珠子,同时释放出可以被吸盘吸收的能量。如果前一颗能量珠的头标记为 m,尾标记为 r,后一颗能量珠的头标记为 r,尾标记为 n,则聚合后释放的能量为 m×r×n(Mars 单位),新产生的珠子的头标记为 m,尾标记为 n

需要时,Mars 人就用吸盘夹住相邻的两颗珠子,通过聚合得到能量,直到项链上只剩下一颗珠子为止。显然,不同的聚合顺序得到的总能量是不同的,请你设计一个聚合顺序,使一串项链释放出的总能量最大。

例如:设 N=44 颗珠子的头标记与尾标记依次为 (2,3)(3,5)(5,10)(10,2)。我们用记号 表示两颗珠子的聚合操作,(jk) 表示第 j,k 两颗珠子聚合后所释放的能量。则第 41 两颗珠子聚合后释放的能量为:

(41)=10×2×3=60

这一串项链可以得到最优值的一个聚合顺序所释放的总能量为:

(((41)2)3)=10×2×3+10×3×5+10×5×10=710

输入格式

第一行是一个正整数 N4N100),表示项链上珠子的个数。第二行是 N 个用空格隔开的正整数,所有的数均不超过 1000。第 i 个数为第 i 颗珠子的头标记(1iN),当 i<N 时,第 i 颗珠子的尾标记应该等于第 i+1 颗珠子的头标记。第 N 颗珠子的尾标记应该等于第 1 颗珠子的头标记。

至于珠子的顺序,你可以这样确定:将项链放到桌面上,不要出现交叉,随意指定第一颗珠子,然后按顺时针方向确定其他珠子的顺序。

输出格式

一个正整数 EE2.1×109),为一个最优聚合顺序所释放的总能量。

样例输入 #1

4 2 3 5 10

样例输出 #1

710

提示

NOIP 2006 提高组 第一题

点击查看代码
#include<iostream> using namespace std; int n, a[205]; long long dp[405][405]; int main() { cin >> n; for(int i = 1; i <= n; i ++) { cin >> a[i]; a[i + n] = a[i]; } for(int len = 2; len <= n + 1; len ++) for(int l = 1; l + len - 1 <= 2 * n; l ++) { int r = l + len - 1; for(int k = l + 1; k <= l + len - 2; k ++) dp[l][r] = max(dp[l][r], dp[l][k] + dp[k][r] + (long long)(a[l] * a[k] * a[r])); } long long res = 0; for(int i = 1; i <= n; i ++) res = max(res, dp[i][n + i]); cout << res << endl; return 0; } 编译结果 compiled successfully time: 5ms, memory: 3920kb, score: 100, status: Accepted > test 1: time: 1ms, memory: 3472kb, points: 10, status: Accepted > test 2: time: 0ms, memory: 3392kb, points: 10, status: Accepted > test 3: time: 0ms, memory: 3448kb, points: 10, status: Accepted > test 4: time: 1ms, memory: 3436kb, points: 10, status: Accepted > test 5: time: 1ms, memory: 3516kb, points: 10, status: Accepted > test 6: time: 0ms, memory: 3692kb, points: 10, status: Accepted > test 7: time: 1ms, memory: 3604kb, points: 10, status: Accepted > test 8: time: 0ms, memory: 3556kb, points: 10, status: Accepted > test 9: time: 1ms, memory: 3816kb, points: 10, status: Accepted > test 10: time: 0ms, memory: 3920kb, points: 10, status: Accepted

3|0B. 开心消消乐

题目描述

A 酱最近在玩开心消消乐,由于是异次元的游戏,所以规则可能和地球上的有所不同。

开心消消乐是一个在大圆环上进行的游戏,环上有若干个宝石,每颗宝石都有自己的积分,由于消消乐是一个三消游戏,我们每次可以挑选其中一个宝石消去,消去宝石的积分为他的积分和左右相邻宝石积分的乘积,比如下左图中,消去 1 的积分就是 1×2×4,如果剩下最后 2 颗宝石,比如下图中,消去 3 的积分是4×3×4,如果剩下最后 1 颗宝石,消去的积分就是他自身的分数。

现在A酱为了通过这一关,需要将宝石消除完,并且获得的积分越大,她的游戏排名就越高,她想请你帮她算算最大得分是多少。

输入数据

共两行,

第一行一个正整数n(1<=n<=500)表示宝石数量

第二行n个正整数ai(1<=i<=n,1<=ai<=100),表示每颗宝石的积分。

输出数据

一行,一个正整数表示获得的最大积分

输入样例1

4 1 2 3 4

输出样例1

84

输入样例2

10 45 29 8 3 32 54 88 68 70 83

输出样例2

2304371
点击查看代码
#include<iostream> using namespace std; int n, a[1005]; long long dp[2005][2005]; int main() { cin >> n; for(int i = 1; i <= n; i ++) { cin >> a[i]; a[i + n] = a[i]; } for(int i = 1; i <= 2 * n; i ++) dp[i][i] = a[i - 1] * a[i] * a[i + 1]; for(int len = 2; len <= n; len ++) { for(int l = 1; l + len - 1 <= 2 * n; l ++) { int r = l + len - 1; for(int k = l; k <= r; k ++) { if(len == n) dp[l][r] = max(dp[l][r], dp[l][k - 1] + dp[k + 1][r] + a[k]); else dp[l][r] = max(dp[l][r], dp[l][k - 1] + dp[k + 1][r] + (long long)(a[l - 1] * a[k] * a[r + 1])); } } } long long res = 0; for(int i = 0; i <= 2 * n; i ++) for(int j = 0; j <= 2 * n; j ++) res = max(res, dp[i][j]); cout << res << endl; return 0; } 编译结果 compiled successfully time: 398ms, memory: 16740kb, score: 100, status: Accepted > test 1: time: 2ms, memory: 6152kb, points: 10, status: Accepted > test 2: time: 47ms, memory: 12396kb, points: 10, status: Accepted > test 3: time: 1ms, memory: 3828kb, points: 10, status: Accepted > test 4: time: 110ms, memory: 16740kb, points: 10, status: Accepted > test 5: time: 0ms, memory: 6128kb, points: 10, status: Accepted > test 6: time: 12ms, memory: 10320kb, points: 10, status: Accepted > test 7: time: 1ms, memory: 3516kb, points: 10, status: Accepted > test 8: time: 61ms, memory: 14556kb, points: 10, status: Accepted > test 9: time: 90ms, memory: 14556kb, points: 10, status: Accepted > test 10: time: 74ms, memory: 14496kb, points: 10, status: Accepted

4|0C. 奶牛的零食

题目描述

约翰经常给产奶量高的奶牛发特殊津贴,于是很快奶牛们拥有了大笔不知该怎么花的钱。为此,约翰购置了 N 份美味的零食来卖给奶牛们。每天约翰售出一份零食。当然约翰希望这些零食全部售出后能得到最大的收益,这些零食有以下这些有趣的特性:

零食按照 1,,N 编号,它们被排成一列放在一个很长的盒子里。盒子的两端都有开口,约翰每天可以从盒子的任一端取出最外面的一个。

与美酒与好吃的奶酪相似,这些零食储存得越久就越好吃。当然,这样约翰就可以把它们卖出更高的价钱。

每份零食的初始价值不一定相同。约翰进货时,第i份零食的初始价值为 Vi1V1000)。

第i份零食如果在被买进后的第 a 天出售,则它的售价是 Vi×a

Vi 的是从盒子顶端往下的第i份零食的初始价值。约翰告诉了你所有零食的初始价值,并希望你能帮他计算一下,在这些零食全被卖出后,他最多能得到多少钱。

输入格式

第一行一个整数 t,代表有 t 组样例,1t50

每组样例:

第一行一个整数 N,代表有 N 件零食,

接下来 2 N+1 行,第 i+1 行代表第 i 份零食的价值 Vi

保证 sum(N)2000

输出格式

每组样例输出约翰最多能卖出多少钱

样例输入 #1

2 5 1 3 1 5 2 2 8 9

样例输出 #1

43 26

提示

第一组样例有 5 件零食,第一天约翰可以选择第 1 件或第 5

最后约翰卖出零食(零食的价值为 1,3,1,5,2)的顺序可以是 1,5,2,3,4,也就是先卖第 1 件,再卖第 5 件,再卖第 2 件,再卖第 3 件,再卖第 4 件。总和 =1×1+2×2+3×3+4×1+5×5=43.

第二组样例有 2 件零食。

点击查看代码
#include<iostream> #include<cmath> #include<string.h> using namespace std; int T, n, dp[2005][2005], v[2005]; int main() { scanf("%d", &T); while(T --) { scanf("%d", &n); for(int i = 1; i <= n; i ++) scanf("%d", &v[i]); for(int i = 1; i <= n; i ++) dp[i][i] = v[i] * n; for(int len = 2; len <= n; len ++) for(int l = 1; l <= n; l ++) { int r = l + len - 1; if(r > n) break; dp[l][r] = max(dp[l][r - 1] + v[r] * (n - len + 1), dp[l + 1][r] + v[l] * (n - len + 1)); } printf("%d\n", dp[1][n]); } return 0; } 编译结果 compiled successfully time: 14ms, memory: 16892kb, score: 100, status: Accepted > test 1: time: 1ms, memory: 3576kb, points: 10, status: Accepted > test 2: time: 0ms, memory: 3788kb, points: 10, status: Accepted > test 3: time: 2ms, memory: 6816kb, points: 10, status: Accepted > test 4: time: 2ms, memory: 4428kb, points: 10, status: Accepted > test 5: time: 0ms, memory: 3964kb, points: 10, status: Accepted > test 6: time: 1ms, memory: 3748kb, points: 10, status: Accepted > test 7: time: 1ms, memory: 15936kb, points: 10, status: Accepted > test 8: time: 1ms, memory: 11532kb, points: 10, status: Accepted > test 9: time: 3ms, memory: 16888kb, points: 10, status: Accepted > test 10: time: 3ms, memory: 16892kb, points: 10, status: Accepted

__EOF__

本文作者So_noSlack
本文链接https://www.cnblogs.com/So-noSlack/p/17570716.html
关于博主:评论和私信会在第一时间回复。或者直接私信我。
版权声明:本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
声援博主:如果您觉得文章对您有帮助,可以点击文章右下角推荐一下。您的鼓励是博主的最大动力!
posted @   So_noSlack  阅读(231)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· winform 绘制太阳,地球,月球 运作规律
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
点击右上角即可分享
微信分享提示