@author: ZZQ

@software: PyCharm

@file: leetcode53_最大子序和.py

@time: 2018/11/26 12:39

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

示例:

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

方法如下:

方法一:暴力遍历法——O(n2)

class Solution():
def maxSubArray(self, nums):
"""
:type nums: List[int]
:rtype: int
"""
nums_len = len(nums)
if nums_len == 0:
return 0
if nums_len == 1:
return nums[0]
max_value = nums[0]
for i in range(nums_len):
temp_max = 0
for j in range(i, nums_len):
temp_max += nums[j] # 大量的重复运算,拖慢速度
max_value = max(max_value, temp_max)
return max_value

方法二:基于方法一避免大量的重复运算 ——O(n2)

class Solution():
def maxSubArray(self, nums):
"""
:type nums: List[int]
:rtype: int
"""
nums_len = len(nums)
temp_sum = [0]*(nums_len+1)
if nums_len == 0:
return 0
if nums_len == 1:
return nums[0]
max_value = nums[0]
for i in range(nums_len): # 基于方法一,避免大量的重复运算
temp_sum[i] = temp_sum[i-1] + nums[i]
for i in range(nums_len):
for j in range(i, nums_len):
temp_max = temp_sum[j] - temp_sum[i-1]
max_value = max(max_value, temp_max)
return max_value

方法三:分治法——O(nlogn)

        将求长度为n的序列中的最大子序和华为分求两个长度为n/2的序列的最大子序和
将当前需要计算最大子串的序列分为两半,
前半段从后往前遍历求最大子串lmax,
后半段从前往后遍历求最大子串rmax,
lmax+rmax就是当前长度为n的序列的最大子串,
然后递归去找长度为n/2的序列的最大子串和,不断求小的序列的子串和。
最后进行比较,得出最大值就是所求结果。
class Solution():
def maxSubArray3(self, nums):
"""
:type nums: List[int]
:rtype: int
"""
nums_len = len(nums)
if nums_len == 0:
return 0
left = 0
right = nums_len - 1
ans = self.maxSubArrayCompute(nums, left, right)
return ans def maxSubArrayCompute(self, nums, left, right):
"""
:type nums: List[int]
:type left: int
:type right: int
:rtype: int
"""
if left > right:
return 0
if left == right:
return nums[left]
middle = (left + right)/2
left_max = self.maxSubArrayCompute(nums, left, middle)
right_max = self.maxSubArrayCompute(nums, middle+1, right)
lmax = nums[middle]
left_sum = nums[middle]
# 保证求得到子序列是连续的(也就是lmax + rmax是一个连续子序列的和)
for i in range(middle-1, left-1, -1): # 前半段逆序加(range(left,right,-1)表示逆序遍历)
left_sum += nums[i]
lmax = max(lmax, left_sum)
rmax = nums[middle+1]
right_sum = nums[middle+1]
for i in range(middle+2, right+1): # 后半段顺序加
right_sum += nums[i]
rmax = max(rmax, right_sum)
return max(rmax+lmax, max(left_max, right_max))

方法四:扫描法——O(n)

        当我们加上一个正数时,和会增加;
当我们加上一个负数时,和会减少。
如果当前得到的和是个负数,那么这个和在接下来的累加中应该抛弃并重新清零,不然的话这个负数将会减少接下来的和。
class Solution():
def maxSubArray(self, nums):
"""
:type nums: List[int]
:rtype: int
"""
nums_len = len(nums)
current = nums[0]
nums_sum = nums[0]
# 我们考虑如果全是负数,那么返回最大的负数,如果最后的和为正,那么就使用扫描法
for i in range(1, nums_len):
if current < 0:
current = nums[i] # 当前数小于0 肯定会舍去(否则将会影响接下来的和),换为下一个数
else:
current += nums[i] # 如果当前数不小于0,那么他会对接下来的和有积极影响
if current > nums_sum:
nums_sum = current # 这里既实现了负数返回最大也实现了扫描法
#这里其实已经隐式的列举了所有可能,保留了所有可能的最大值
return nums_sum

方法五:动态规划——O(n)

        假设sum[i]为以i结尾的连续子串的和. 假设对于元素i,所有以它前面的元素结尾的子串和都已经求得,
那么以第i个元素结尾且和最大的连续子串实际上,
1)要么是以第i-1个元素结尾且和最大的连续子数组加上这个元素,
2)要么只是第i个元素,即sum[i]
所以当前的最大连续子串和的计算方法是max(sum[i-1]+nums[i], nums[i])
这就是需要我们判断sum[i-1]是否大于0。
由于每次运算只需要前一次的结果,因此算法的时间和空间复杂度都很小。
class Solution():
def maxSubArray5(self, nums): # 动态规划
"""
:type nums: List[int]
:rtype: int
"""
nums_len = len(nums)
if nums_len == 0:
return 0
if nums_len == 1:
return nums[0]
nums_sum = nums[0]
pre = nums[0]
for i in range(1, nums_len):
if pre > 0:
pre += nums[i]
else:
pre = nums[i]
nums_sum = max(nums_sum, pre) return nums_sum

Leetcode——53.最大子序和的更多相关文章

  1. LeetCode 53. 最大子序和(Maximum Subarray)

    53. 最大子序和 53. Maximum Subarray 题目描述 给定一个整数数组 nums,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和. LeetCode53. M ...

  2. Java实现 LeetCode 53 最大子序和

    53. 最大子序和 给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和. 示例: 输入: [-2,1,-3,4,-1,2,1,-5,4], 输出: 6 ...

  3. Leetcode之动态规划(DP)专题-53. 最大子序和(Maximum Subarray)

    Leetcode之动态规划(DP)专题-53. 最大子序和(Maximum Subarray) 给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和. ...

  4. leetcode 120. 三角形最小路径和 及 53. 最大子序和

    三角形最小路径和 问题描述 给定一个三角形,找出自顶向下的最小路径和.每一步只能移动到下一行中相邻的结点上. 例如,给定三角形: [ [2], [3,4], [6,5,7], [4,1,8,3] ] ...

  5. 1. 线性DP 53. 最大子序和.

    53. 最大子序和. https://leetcode-cn.com/problems/maximum-subarray/ func maxSubArray(nums []int) int { dp ...

  6. 53. 最大子序和(剑指 Offer 42)

    53. 最大子序和(剑指 Offer 42) 知识点:数组:前缀和:哨兵:动态规划:贪心:分治: 题目描述 输入一个整型数组,数组中的一个或连续多个整数组成一个子数组.求所有子数组的和的最大值. 要求 ...

  7. 【LeetCode】53.最大子序和

    最大子序和 给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和. 示例: 输入: [-2,1,-3,4,-1,2,1,-5,4], 输出: 6 解释: ...

  8. Leetcode题目53.最大子序和(动态规划-简单)

    题目描述: 给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和. 示例: 输入: [-2,1,-3,4,-1,2,1,-5,4],输出: 6解释: 连 ...

  9. leetcode之53.最大子序和

    题目详情 给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和. 示例: 输入: [-2,1,-3,4,-1,2,1,-5,4], 输出: 6 解释: ...

随机推荐

  1. do-while语句及for语句(初学者)

    1.do-while语句的一般形式为: do 语句 while(表达式): 这个循环与while循环的不同在于:它先执行循环中的语句,然后再判断这个表达式是否为真,如果为真则继续循环:如果为假,则中止 ...

  2. python分包写入文件,写入固定字节内容,当包达到指定大小时继续写入新文件

    第6行通过 for 循环控制生成 .log 文件的数量 第8行,如果该文件存在时先进行清空,然后再进行写入操作 第13行,将文件大小的单位转为MB 第14行,如果文件大小超过1MB时,跳出当前循环,重 ...

  3. 线程间的通信_多生产者多消费者问题_JDK1.5新特性_Lock

    对于同步代码块,对于锁的操作是隐式的但是在JDK1.5之前的这种方法效率有点低,判断会很多,后面升级之后有新的解决方案 jdk1.5以后将同步和锁封装成了对象,并将操作锁的隐式方式定义到了该对象中,将 ...

  4. 解决HTTP status code is not handled or not allowed

    /Books/>: HTTP status code is not handled or not allowed 2017-11-04 17:21:38 [scrapy.spidermiddle ...

  5. Redis学习笔记--Redis数据过期策略详解==转

    本文对Redis的过期机制简单的讲解一下 讲解之前我们先抛出一个问题,我们知道很多时候服务器经常会用到redis作为缓存,有很多数据都是临时缓存一下,可能用过之后很久都不会再用到了(比如暂存sessi ...

  6. Qt+Qgis二次开发:地理实体抽象

    1  概述 地理实体抽象是指点.线.面及其组合而成的,用于描述实际地物的数据结构. 其中包含几何实体和属性数据. GIS中进行几何操作,以各种实体类为基础进行操作. 在OGC中,地理实体可以由WKT表 ...

  7. vue filters过滤器

    vue filters过滤器 vue.js允许我们自定义过滤器,可被使用于一些常见的文本格式化,过滤器可以用在两个地方,双花括号插值和 v-bind表达式.最常见的就是双花括号插值. 比如如下代码:{ ...

  8. ESP32 environment ubuntu

    ubuntu官方用的是16.04版本的,阿里源下载地址:http://mirrors.aliyun.com/ubuntu-releases/16.04/     用官方版本的好处就是省的搞一堆各种错误 ...

  9. 10-(基础入门篇)MQTT介绍

    https://www.cnblogs.com/yangfengwu/p/9953920.html 看到这个项目第一想法肯定需要一个服务器,所有的wifi设备和手机都去连接这个服务器,然后服务器进行信 ...

  10. sql server使用的相关基础知识

    1.表的管理--表和列的命名 必须以字母开头 长度不能超过128字符 不要使用sql server的保留字 只能使用如下字符A-Z,a-z,0-9,$,#,_等等 2.表的管理--支持的数据类型 字符 ...