四种方法求解最大子段和问题

article/2025/11/7 0:58:44

题目描述

在这里插入图片描述
给定一段长度为n的序列,我们需要找到其中一个连续的子段,使这个子段中各个元素加和最大,如果这个数组中全为负整数,我们就定义这个子段和为0.

题目分析

首先我们的目的是找一个局部的子段但加和是全局最大,所以我们可以很自然地想到,直接暴力法遍历求解即可。
方法一.暴力法求解
我们设置一个i,从第1个位置开始遍历,这个i表示的是我们最大子段的起始位置,然后我们从j=i往后接着遍历,j表示的是我们最大子段的末尾位置。接着从初始位置到末尾位置的元素累加求和,不断更新最大值即可。
暴力法代码

#include <iostream>
using namespace std;
int Maxsum(int n,int *a,int &besti,int &bestj)
{int sum = 0;for(int i = 1;i <= n;i++)//个人编程习惯,数组第一个位置从1开始{for(int j = i;j <= n;j++){int thissum = 0;//当前最大值,我们用它来更新sumfor(int k = i;j <= j;k++){thissum += a[k];}if(thissum > sum){sum = thissum;besti = i;//最优区间的开头bestj = j;//最优区间的结尾}}}return sum;
}

暴力法通常都可以解决问题,但也绝不是最好的方法,它的时间复杂度是极大的。大家可以看到,我们的代码中用了三层嵌套循环,所以这个方法的时间复杂度是O(n^3),空间复杂度是O(1)
方法二.暴力法的改良
暴力法的时间复杂度过高,我们可以基于传统的暴力法再优化一点。就是在我们移动末尾区间的时候,边移动边求和。不用等到末尾区间固定了之后再遍历求和。
暴力法改良代码

#include <iostream>
using namespace std;
int Maxsum(int n,int *a,int &besti,int &bestj)
{int sum = 0;for(int i = 1;i <= n;i++)//个人编程习惯,数组第一个位置从1开始{int thissum = 0;for(int j = i;j <= n;j++){thissum += a[j];if(thissum > sum){sum = thissum;besti = i;//最优区间的开头bestj = j;//最优区间的结尾}}}return sum;
}
我们减少了一层循环,所以时间复杂度减少为了O(n^2),空间复杂度为O(1)

方法三.分治法
分治法是解决数组问题很常用的一种方法,我们对分治法情有独钟又敬而远之。其实分治法的本质就是递归,只要想着,我 们就负责把问题给你分开,分开的问题就留给你计算机自己处理。
分治法解题有三种情况:
1,这个最大子段在我们数组的左侧
2.这个最大子段在我们数组的右侧
3.这个最大子段跨过了左右两侧,在中间最大。
下面我们会分别讨论这三种情况。
第一种和第二种我们很简单,我们只需要简单地递归来解题,将两个子问题递归解出。分开的位置就是我们的中心位置。
第三种情况,我们假设跨过中心的子段在左侧的最大值为s1,在右侧的最大值为s2.则这个完整子段的最大值就是s1+s2。看吧,我们又把问题分成了两个。分别求解就好了。
分治法代码

#include <iostream>
using namespace std;
int Maxsubsum(int *a,int left,int right)
{int sum = 0;if(left == right)return a[left] > a[right] ? a[left] : a[right];else{int center = (left + right) / 2;int leftsum = Maxsubsum(a,left,center);int rightsum = Maxsubsum(a,center+1,right);int s1 = 0;int lefts = 0;//从中心向左侧遍历求最大for(int i = center;i >= left;i--){lefts += a[i];if(lefts>s1)s1 = lefts;//为了防止我们加入了负数,所以每次都应该执行这部操作来判断}//对于右侧同理int s2 = 0;int rights = 0;//从中心向右侧遍历求最大for(int j = center+1;j <= right;j++){rights += a[j];if(rights > s2)s2 = rights;}sum = s1 + s2;if(sum < leftsum)sum = leftsum;if(sum < rightsum)sum = rightsum;}return sum;
}
//但是这个方法我们也可以找到最优的区别,只需要稍微修改下上面的程序
int Maxsum(int n,int *a)
{return Maxsubsum(a,1,n);
}

数组被我们不断二分,分的次数为k,所以n = 2^k,遍历的时间复杂度为n,所以这个方法的时间复杂度为O(nlogn)
动态规划法
接下来就是最核心的方法了,其实这道题相信很多朋友看到了第一时间就会想到动态规划法,没错,这确实也是最简单的方法。
在这里插入图片描述
我们先开辟一个新的数组b,b[k]存储的是遍历到a[k]时的最大子段和。我们假设现在访问到了第j个位置,则a[j]位置的最大子段和应该是b[j],所以我们应该更新b[j]=b[j-1]+a[j]
但是如果b[j-1]<0的话,我们就不做上述更新了,因为a[j]+b[j-1]一定小于a[j],所以我们就让b[j]=a[j]即可。
动态规划法代码

#include <iostream>
using namespace std;
int Maxsum(int n,int *a)
{int sum = 0,b = 0;for(int i = 1;i <= n;i++){if(b > 0)b += a[i];elseb = a[i];if(b > sum)sum = b;}return sum;
}

动态规划法我们只遍历了一次数组,所以时间复杂度为O(n).

总结

对于数组问题,无论是一维或者二维,求最大或最小值一定会涉及到一个回溯或者动态规划问题,我们需要回头找到一个最值,或者直接干脆将前面的最值存起来。从而可以简化我们的计算。但是有的时候单纯运用动态规划法反而会造成更大的时间浪费,所以可能还需要一些小小的优化。具体例子可见我的另一篇博客:最长单调递增子序列


http://chatgpt.dhexx.cn/article/FIeDAWeI.shtml

相关文章

最大子段和问题(分治法和动态规划)

什么是最大子段和&#xff0c;通俗点讲&#xff1a; 最大子段和就是给了一些数&#xff0c;然后你从中找了几个连续的数&#xff0c;这组连续的数的和比任意一组连续的数的和都大&#xff0c;那么你找的这几个连续的数的和就是这些数的最大子段和。 通俗的听不懂你就看这里&am…

最大子段求和

3种算法&#xff1a;最大子段求和 一、问题分析 问题&#xff1a;给定有n个整数(可能为负整数)组成的序列 a 1 , a 2 , . . . , a n , a_1,a_2,...,a_n, a1​,a2​,...,an​,求该序列连续的子段和的最大值。 如果该子段的所有元素和是负整数时定义其最大子段和为0。 简易算法…

最大子段和问题(3种方法)

给定由n个整数(可能为负整数)组成的序列a1&#xff0c;a2&#xff0c; a3… &#xff0c; an&#xff0c; 寻找它的某个连续子段&#xff0c;使得其和最大。例如( -2,11,-4,13,-5,-2 )最大子段是{ 11,-4,13 }其和为20。 1、最大字段和问题的简单算法 (1)枚举法求解&#xff1…

最大子段和(动态规划算法)

最大子段和&#xff08;动态规划算法&#xff09; 文章目录 最大子段和&#xff08;动态规划算法&#xff09;一、思路二、伪代码三、C代码四、输入实例 一、思路 D[i]表示从i开始的最大字段和。&#xff08;但我们不是从前往后找字段结束位置&#xff09; 根据递推公式&#x…

最大子段和——用蛮力算法,分治策略,动态规划算法三种求法(C语言)

目录 一、题目 二、算法求解 1、蛮力算法 伪代码 算法分析 程序 2、分治策略 伪代码 算法分析 程序 3、动态规划算法 伪代码 算法分析 程序 一、题目 设A<a1,a2,...,an>是n个整数的序列&#xff0c;称<ai,....,aj>为该序列的连续子序列&#xff0c;其…

归并排序 java实现_java实现归并排序

归并排序 归并排序&#xff0c;指的是将两个已经排序的序列合并成一个序列的操作。 归并操作的过程如下&#xff1a; 申请空间&#xff0c;使其大小为两个已经排序序列之和&#xff0c;该空间用来存放合并后的序列 设定两个指针&#xff0c;最初位置分别为两个已经排序序列的起…

Java八大算法:归并排序

一、什么是归并排序&#xff1f; 1.概念 归并排序&#xff08;Merge sort&#xff09;是建立在归并操作上的一种有效的排序算法&#xff0c;归并排序对序列的元素进行逐层折半分组&#xff0c;然后从最小分组开始比较排序&#xff0c;合并成一个大的分组&#xff0c;逐层进行…

归并排序(Java代码实现)

基本思想&#xff1a; 将两个或两个以上的有序表合并成一个有序表的过程。常用的归并为2-路归并&#xff0c;就是将两个有序表合为一个有序表。 过程&#xff1a; 先来看一张示意图&#xff1a; 可以看出&#xff0c;归并排序分为分解和合并两个步骤。 分解就是将原数组分解…

归并排序(Java实现)

归并排序&#xff08;Merge Sort&#xff09;是建立在归并操作上的一种有效&#xff0c;稳定的排序算法&#xff0c;该算法是采用分治法&#xff08;Divide and Conquer&#xff09;的一个非常典型的应用。将已有序的子序列合并&#xff0c;得到完全有序的序列&#xff1b;即先…

归并排序java详解

import java.util.Arrays;public class Mergesort {public static void main(String[] args) {int arr[]{8,4,5,7,1,3,6,2};int temp[]new int[arr.length];//归并排序需要一个额外空间mergeSort(arr,0,arr.length-1,temp);System.out.println("归并排序后" Arrays.t…

java归并排序(含归并排序代码)

目录 一&#xff1a;归并排序的思想 二&#xff1a;归并排序代码 三&#xff1a;归并排序结果​ 一&#xff1a;归并排序的思想 归并排序&#xff08;Merge Sort&#xff09; 当我们要排序这样一个数组的时候&#xff0c;归并排序法首先将这个数组分成一半。如图&#xff1a…

归并排序java代码实现

归并排序,是一种分治算法。利用递归&#xff0c;将一个大的数据集合分解成小的子集合。将子集合排好序后&#xff0c;再合并起来。归并排序不是原地排序算法,因为它使用到了临时空间&#xff0c;这也是归并排序没有快速排序应用广泛的主要原因&#xff0c;虽然归并排序的时间复…

分治算法实现经典归并排序java实现

目录 1.什么是分治算法 分治法 基本思想 2.分治算法的体现:归并排序 归并排序 基本思想 3.代码实现 1.什么是分治算法 分治法 分治法&#xff0c;字面意思是“分而治之”&#xff0c;就是把一个复杂的1问题分成两个或多个相同或相似的子问题&#xff0c;再把子问题分成…

归并排序 Java实现 简单易懂

归并排序 归并排序采用的是分治(divide-and-conquer)法思想。 1.基本思想&#xff1a; 将待排序元素分成大小大致相同的2个子集合&#xff0c;分别对2个子集合进行排序&#xff0c;最终将排好序的子集合合并成为所要求的排好序的集合。 2.执行过程&#xff1a; 3.时间复杂度…

归并排序java版

归并排序java版 归并排序java版 归并排序java版 好长时间没写过归并排序&#xff0c;在学习并发中又遇到了一个归并排序的demo&#xff0c;于是就想试试自己还能不能写出来&#xff0c;结果没写出来…&#xff0c;看了一些文章后&#xff0c;整理了一下思路&#xff0c;把归并…

归并排序(Java)

归并排序 归并排序&#xff08;MERGE-SORT&#xff09;是建立在归并操作上的一种有效的排序算法,该算法是采用分治法&#xff08;Divide and Conquer&#xff09;的一个非常典型的应用。将已有序的子序列合并&#xff0c;得到完全有序的序列&#xff1b;即先使每个子序列有序&a…

JAVA编程----归并排序

一、概念及其介绍 归并排序&#xff08;Merge sort&#xff09;是建立在归并操作上的一种有效、稳定的排序算法&#xff0c;该算法是采用分治法(Divide and Conquer&#xff09;的一个非常典型的应用。将已有序的子序列合并&#xff0c;得到完全有序的序列&#xff1b;即先使每…

归并排序java(内附超详解图文讲解)

目录 归并排序介绍: 归并排序图解&#xff1a; 示意图1 示意图2 归并排序详细步骤&#xff1a; 1.组合 2.分组 完整代码 归并排序介绍: 归并排序(MERGE-SORT)是利用归并的思想实现的排序方法&#xff0c;该算法采用经典的分治(divide- and-conquer)策略(分治法将问题分…

java实现归并排序(详解)

主要思想 归并排序和快速排序都是基于分而治之的算法思想。 归并排序先将待排序的数组不断拆分&#xff0c;直到拆分到区间里只剩下一个元素的时候。不能再拆分的时候。这个时候我们再想办法合并两个有序的数组&#xff0c;得到长度更长的有序数组。当前合并好的有序数组为下…

Launcher3--抽屉

抽屉是用来放置安卓手机中所有需要显示到Launcher上的(当然也可以进行过滤&#xff0c;将不想显示的隐藏起来)应用和小部件&#xff0c;启动应用、添加快捷方式到桌面、卸载等。之前也提到过&#xff0c;有些Launcher是没有抽屉的&#xff0c;如MIUI的Launcher。在Launcher3中&…