题目描述:
一个有 n 个元素的数组,这 n 个元素既可以是正数也可以是负数,数组中连续
的一个或多个元素可以组成一个连续的子数组,一个数组可能有多个这种连续的子数组,求子数组的最大值。例如,对于数组 [1,-2,4,8,-4,7,-1,-5] 而言,其最大和的子数组为 [4,8,-4,7],最大值为 15。
方法:
- 蛮力法
- 重复利用已经计算的子数组和
- 动态规划
- 优化的动态规划
1.蛮力法
找出所有的子数组,然后求出子数组的和,在所有子数组的和中取最大值。
代码实现:
#!/usr/bin/env python3 # -*- coding: utf-8 -*- # @Time : 2020/1/29 21:59 # @Author : buu # @Software: PyCharm # @Blog :https://blog.csdn.net/weixin_44321080 def maxSubArrSum(arr): if arr == None or len(arr) <= 0: print('参数不合理!') return thisSum = 0 maxSum = 0 i = 0 while i < len(arr): j = i while j < len(arr):# j 控制连续子数组包含的元素个数 thisSum = 0 k = i # k 表示连续子数组开始的下标 while k < j: thisSum += arr[k] k += 1 if thisSum > maxSum: maxSum = thisSum j += 1 i += 1 return maxSum if __name__ == '__main__': arr = [1, -2, 4, 8, -4, 7, -1, -5] print('1 max sub array sum:', maxSubArrSum(arr))
结果:
算法性能分析:
这种方法的时间复杂度为O(n3);
2.重复利用已经计算的子数组和
由于 sum[i,j] = sum[i,j-1] + arr[j],在计算 sum[i,j] 的时候可以使用前面已计算出的 sum[i,j-1] 而不需要重新计算,采用这种方法可以省去计算 sum[i,j-1] 的时间,从而提高效率。
代码实现:
#!/usr/bin/env python3 # -*- coding: utf-8 -*- # @Time : 2020/1/30 10:53 # @Author : buu # @Software: PyCharm # @Blog :https://blog.csdn.net/weixin_44321080 def maxSubArrSum(arr): if arr == None or len(arr) <= 0: print('参数不合理!') return maxSum = -2 ** 31 i = 0 while i < len(arr): # i: 0~7 sums = 0 j = i while j < len(arr): # j: 0~7 sums += arr[j] # sums 重复利用 if sums > maxSum: # 每加一次就判断一次 maxSum = sums j += 1 i += 1 return maxSum if __name__ == '__main__': arr = [1, -2, 4, 8, -4, 7, -1, -5] print('2 max sub array sum:', maxSubArrSum(arr))
结果:
算法性能分析:
使用了双重循环,时间复杂度为O(n2);
3.动态规划
首先可以根据数组最后一个元素 arr[n-1] 与最大子数组的关系分为以下三种情况讨论:
(包含或不包含,包含的话肯定以最后一个元素结尾;不包含的话,或者最后一个元素单独构成最大子数组,或者转换为求 arr[1…n-2] 的最大子数组)
(1) 最大子数组包含 arr[n-1],即最大子数组以 arr[n-1] 结尾;
(2) arr[n-1] 单独构成最大子数组;
(3) 最大子数组不包含 arr[n-1],那么求 arr[1…n-1] 的最大子数组可以转换为求 arr[1…n-2] 的最大子数组。
所以有:
代码实现:
#!/usr/bin/env python3 # -*- coding: utf-8 -*- # @Time : 2020/1/30 11:19 # @Author : buu # @Software: PyCharm # @Blog :https://blog.csdn.net/weixin_44321080 def maxSubArrSum(arr): if arr == None or len(arr) <= 0: print('参数不合理!') return n = len(arr) End = [None] * n # End[i] 表示包含 arr[i] 的最大子数组和 All = [None] * n # All[i] 表示最大子数组和 End[n - 1] = arr[n - 1] All[n - 1] = arr[n - 1] End[0] = All[0] = arr[0] i = 1 while i < n: End[i] = max(End[i - 1] + arr[i], arr[i]) # i=1时若arr[0]<0,则从arr[1]重新开始 All[i] = max(End[i], All[i - 1]) i += 1 return All[n - 1] if __name__ == '__main__': arr = [1, -2, 4, 8, -4, 7, -1, -5] print('3 max sub array sum:', maxSubArrSum(arr))
结果:
算法性能分析:
时间复杂度为O(n);
由于额外申请了两个数组,所以空间复杂度为O(n);
4.优化的动态规划
方法3中每次其实只用到了 End[i-1] 与 All[i-1] ,而不是整个数组中的值,所以可以定义两个变量来保存 End[i-1] 与 All[i-1] 的值,并且可以反复利用。
代码实现:
#!/usr/bin/env python3 # -*- coding: utf-8 -*- # @Time : 2020/1/30 11:55 # @Author : buu # @Software: PyCharm # @Blog :https://blog.csdn.net/weixin_44321080 def maxSubArrSum(arr): if arr == None or len(arr) <= 0: print('参数不合理!') return nAll = arr[0] # 最大子数组和 nEnd = arr[0] # 包含最后一个元素的最大子数组和 i = 1 while i < len(arr): nEnd = max(nEnd + arr[i], arr[i]) nAll = max(nEnd, nAll) i += 1 return nAll if __name__ == '__main__': arr = [1, -2, 4, 8, -4, 7, -1, -5] print('4 max sub array sum:', maxSubArrSum(arr))
结果:
算法性能分析:
时间复杂度为O(n);
空间复杂度为O(1);
引申:
在知道了子数组的最大值后,如何确定最大子数组的和?
思路:
代码实现:
#!/usr/bin/env python3 # -*- coding: utf-8 -*- # @Time : 2020/1/30 12:01 # @Author : buu # @Software: PyCharm # @Blog :https://blog.csdn.net/weixin_44321080 class Test: def __init__(self): self.begin = 0 # 记录最大子数组起始位置 self.end = 0 # 记录最大子数组结束位置 def maxSubArrSum(self, arr): n = len(arr) maxSum = -2 ** 31 # 子数组最大值 nSum = 0 # 包含子数组最后一位的最大值 nStart = 0 i = 0 while i < n: if nSum < 0: nSum = arr[i] nStart = i else: nSum += arr[i] if nSum > maxSum: maxSum = nSum self.begin = nStart self.end = i i += 1 return maxSum def getBegin(self): return self.begin def getEnd(self): return self.end if __name__ == '__main__': arr = [1, -2, 4, 8, -4, 7, -1, -5] t = Test() print('连续最大和为:', t.maxSubArrSum(arr)) print('begin at ', t.getBegin()) print('end at ', t.getEnd())
结果:
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件! 如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。
更新日志
- 明达年度发烧碟MasterSuperiorAudiophile2021[DSF]
- 英文DJ 《致命的温柔》24K德国HD金碟DTS 2CD[WAV+分轨][1.7G]
- 张学友1997《不老的传说》宝丽金首版 [WAV+CUE][971M]
- 张韶涵2024 《不负韶华》开盘母带[低速原抓WAV+CUE][1.1G]
- lol全球总决赛lcs三号种子是谁 S14全球总决赛lcs三号种子队伍介绍
- lol全球总决赛lck三号种子是谁 S14全球总决赛lck三号种子队伍
- 群星.2005-三里屯音乐之男孩女孩的情人节【太合麦田】【WAV+CUE】
- 崔健.2005-给你一点颜色【东西音乐】【WAV+CUE】
- 南台湾小姑娘.1998-心爱,等一下【大旗】【WAV+CUE】
- 【新世纪】群星-美丽人生(CestLaVie)(6CD)[WAV+CUE]
- ProteanQuartet-Tempusomniavincit(2024)[24-WAV]
- SirEdwardElgarconductsElgar[FLAC+CUE]
- 田震《20世纪中华歌坛名人百集珍藏版》[WAV+CUE][1G]
- BEYOND《大地》24K金蝶限量编号[低速原抓WAV+CUE][986M]
- 陈奕迅《准备中 SACD》[日本限量版] [WAV+CUE][1.2G]