数学考试的拓展题

数学考试的拓展题:

简单题意:将n个数的数列分为m个不相交的连续子序列,求这m个子段的最大值。


首先容易考虑用dp优化暴力做法。
我们考虑dp[i][j] 表示以j结尾且前j项被分为i段的最大值
那么有两种情况:
1.dp[i][j] = dp[i - 1][k] + a[j] (表示的是第j个元素单独在一组)
2.dp[i][j] = dp[i][j - 1] + a[j] (表示的是第j个元素与前面在同一组内)
于是转移方程就是 dp[i][j] = max(dp[i - 1][k], dp[i][j - 1]) + a[j]; 其中k=i-1,i,...,j-1。
复杂度是O(n^3), 显然对于n<=5000的复杂度是不够的。
接下来我们就需要优化时间,由转移方程可知如果我们枚举i,那么i这一层只和i层和i-1层有关,那么我们可以从小到大枚举j,并且用一个数组ma[n]去保存i-1层的dp[i - 1][k]的数据。ma[j - 1]存的是max(dp[i - 1][i - 1], dp[i - 1][i]...,dp[i - 1][j - 1])

for (int i = 1; i <= m; i++){
    maxx = -inf;
    for (int j = i; j <= n; j++){
        dp[i][j] = max(dp[i][j - 1], ma[j - 1]) + a[j];
        ma[j - 1] = maxx;
        maxx = max(dp[i][j], maxx);
    }
}

不难发现dp的第一维可以省略。可以优化到下面的代码。

for (int i = 1; i <= m; i++){
    maxx = -inf;
    for (int j = i; j <= n; j++){
        dp[j] = max(dp[j - 1], ma[j - 1]) + a[j];
        ma[j - 1] = maxx;
        maxx = max(dp[j], maxx);
    }
}
全部评论

相关推荐

03-04 17:07
南昌大学 Java
点赞 评论 收藏
分享
牛客44320985...:你的当务之急是把这个糖的要死的沟槽ide主题改了
点赞 评论 收藏
分享
多多啊&nbsp;多多啊&nbsp;上来四道算法题算法题直播排序,整体比较简单把对象写出来,然后比较规则写明白就OK了。唯一一道A100%的电车充电如何最省钱,到目的地如何充电的钱最少,路上有充电站,每个电站价格不一样。用了DP来做,但感觉是贪心的样子,最后没招了,把不能到的情况给干了出来,过了8%日志分析纠错,滑动窗口,但我最后结果永远少一,过了15%没看,力竭了燃尽了多多&nbsp;以后牛客不用后台找我了,笔试夯爆了
淮竹c:不好意思,打扰大家🙏我是一个拼多多骑手,小电驴的最大电量为C,我的最大电量有1e9这么promax😭😭😭需要从x=0处走到x=L,L足足有1e9那么长处,途中有n个充电站,🙏🙏每个充电站的距离和电价分别为di和pi,初始电量是满的😭😭😭请告诉我到达终点最少要花多少钱😭😭😭求求大家把这些钱转给我
查看2道真题和解析
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务