python 最大连续子数组的和

抛出问题:

  求一数组如 l = [0, 1, 2, 3, -4, 5, -6],求该数组的最大连续子数组的和 如结果为[0,1,2,3,-4,5] 的和为7

问题分析:

  这个问题很简单,直接暴力法,上代码。

# -*- coding:utf-8  -*-
# 日期:2018/6/9 7:46
# Author:小鼠标

# 最大连续子数组的和
l = [0, 1, 2, 3, -4, 5, -6]
# 暴力求解
def violence(l = []):
    maxVal = 0
    x,y=0,0
    for i in range(0,len(l)+1):
        for j in range(0,len(l)+1):
            res = sum(l[i:j])
            if res > maxVal:
                maxVal = res
                x = i
                y = j
    return maxVal,x,y
maxVal, x, y = violence(l)
print(maxVal,(x,y))

分治法:

  关键是暴力法的时间复杂度太高,所以就在原有的基础上做了进一步的提升--分治法。

  所谓分治法就是将原有的列表一分为二,那么最大的子列表只有三种情况:

  1、最大子列表完全在左边

  2、最大子列表完全在右边

  3、最大子列表跨立在中间

  所以我们分情况讨论,求出答案。这种方法一定程度的降低了时间复杂度,从之前的n^2降到了(n/2)^2 + 2n

# -*- coding:utf-8  -*-
# 日期:2018/6/9 7:46
# Author:小鼠标

# 最大连续子数组的和
l = [0, 1, 2, 3, -4, 5, -6]
#暴力求解
def violence(l = []):
    maxVal = 0
    x,y=0,0
    for i in range(0,len(l)+1):
        for j in range(0,len(l)+1):
            res = sum(l[i:j])
            if res > maxVal:
                maxVal = res
                x = i
                y = j
    return maxVal,x,y
#分治法 想左扫 向右扫,求出两边的最大值
def left_or_right(l):
    maxVal = 0
    term = 0
    for i in l:
        term += i
        if maxVal < term:
            maxVal = term
    return maxVal
def Separate():
    middle = int(len(l)/2)
    l1 = l[0:middle]
    l2 = l[middle:len(l)]
    #左半部分
    maxVal1,x1,y1 = violence(l1)
    #右半部分
    maxVal2,x2,y2 = violence(l2)
    #跨立在中间
    max_right = left_or_right(l2)
    max_left = left_or_right(l1[::-1])
    maxVal3 = max_right + max_left
    return max(maxVal1,maxVal2,maxVal3)
val = Separate()
print(val)

动态规划:

  即便是分治法,时间复杂度还是太高,不满足生产的需求,所以如果说只求最大子序列的和的值而不去追求最大子序列本身,我们又引出一个方法--动态规划

  这种方法的时间复杂是是线性的,极大的降低了。

# -*- coding:utf-8  -*-
# 日期:2018/6/9 8:38
# Author:小鼠标

def function(lists):
    max_sum = lists[0]
    pre_sum = 0
    for i in lists:
        # 因为最大子列表一定是从一个非0的数开始的(假定列表中有正数有负数)
        # 所以就可以暂时筛选调小于0的数,即便列表全是负数,那么最大的子列表肯定是负数最大的一个
        if pre_sum < 0:
            pre_sum = i
        else:
            pre_sum += i
        if pre_sum > max_sum:
            max_sum = pre_sum
    return max_sum
lists = [0, 1, 2, 3, -4, 5, -6]
print(function(lists))

 

转载于:https://www.cnblogs.com/7749ha/p/9162194.html

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值