最大子序列和(动态规划)

给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。

示例:

输入: [-2,1,-3,4,-1,2,1,-5,4],
输出: 6
解释: 连续子数组 [4,-1,2,1] 的和最大,为 6。

根据题意,我们必须明确,答案是一个 连续的最大和子数组,它和最大上升子序列是不同的。

一般来说过,只要一旦判断题目是有关动态规划的题,第一步就是要找到他的最优子结构,至于怎么找到最优子结构,只能刷题了,多刷点题目。

用数组d[i] 来保存 当前最大的连续子数组,算法的思想大体是这样的,循环遍历每个数,然后每次检验d[i-1] 是否大于零,只要大于零就    d[i] = d[i-1]+nums[i] ,如果d[i-1]<0 ,那么直接d[i]=nums[i]

算法核心: d[i] = d[i-1]>=0 ? d[i-1]+nums[i] : nums[i]

我在这里下标按 1...n 比较好描述

按照上面算法核心走一遍

[-2]                                           d[1] = -2;   这步是直接初始化,在只有一个数的时候,他就是最大连续的最大和子数组

[-2,1]                                       d[2] = 1;  d[1] =-2 <0 所以前面的最大连续和是负数,加上它们肯定会变小,所以直接不要它

[-2,1,-3]                                   d[3] = d[2]+nums[3] =-2; d[2] =1>=0 所以在i之前的最大连续是正数,肯定要加上它

[-2,1,-3,4]                                d[4] = 4; d[3] < 0

[-2,1,-3,4,-1]                           d[5] = d[4] +nums[5] = 3 ;  d[4]>0

[-2,1,-3,4,-1,2]                        d[6] = d[5]+nums[6] =5; d[5]>0

[-2,1,-3,4,-1,2,1]                     d[7] = d[6]+nums[7] =6; d[6]>0

[-2,1,-3,4,-1,2,1,-5]                d[8] = d[7]+nums[8] =1;d[7]>0

[-2,1,-3,4,-1,2,1,-5,4]             d[9] = d[8]+nums[9] =5; d[8]>0

在这里我们只需要遍历d[i]数组中最大数数即可,或者每一遍历的使用m来记录d[i] 的最大值,具体算法过程如下

#include <iostream>
#include <math.h>
using namespace std;
int main()
{
    //d 用来保存最大连续子数组的值
    //m 记录d数组中的最大值
    int n,nums[100],d[100],m;
    while(cin>>n)
    {
        for(int i=1;i<n;i++)
        {
            cin>>nums[i];
        }
        m=d[0] = nums[0];    // 初始化第一个 d[0]
        for(int i=1;i<n;i++) // 遍历从第二数开始
        {
            if(d[i-1]<0)     // i 之前的最大连续子数组小于零,则抛弃它,自己从开始
                d[i] = nums[i];
            else             // i 之前的最大连续子数组不小于零,则加上他们
                d[i] = nums[i]+d[i-1];
            m = max(d[i],m);  // 保存d[i] 中的最大值
        }
        cout<<m<<endl;
    }
    return 0;
}

上面代码是我直接手敲的,没过编译器,但思想不会错了,  这个时间复杂度为 O(n)

 

 

                              

 

  • 15
    点赞
  • 58
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 12
    评论
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

memory_cood

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值