java动态规划求最大子段和_动态规划-最大子段和

article/2025/11/7 1:03:48

2018-01-14 21:14:58

一、最大子段和问题

问题描述:给定n个整数(可能有负数)组成的序列a1,a2,...,an,求该序列的最大子段和。如果所有整数都是负数,那么定义其最大子段和为0。

方法一、最大子段和的简单算法

显然可以在O(n^2)的时间复杂度上完成这个问题。但是是否可以对算法进行优化呢?答案是肯定的。

方法二、分治算法

朴素的分法是二分,问题就是如何merge,在merge的时候,因为已经确定了会包含边界点,所以可以在O(n)的时间复杂度上完成merge时的最大子段和。

因此分治公式是:T(n) = 2T(n/2)+O(n)

根据主定理可以算得,分治算法的时间复杂度为O(nlgn)。

int maxSubSum(int[] a,int L,int R){

if(L == R) return a[L] > 0 ? a[L] : 0;

else {

int mid = L + (R - L) / 2;

int sumL = maxSubSum(a, L, mid);

int sumR = maxSubSum(a, mid + 1, R);

int tmpL = 0;

int tmpR = 0;

int sum = 0;

for (int i = mid; i >=0 ; i--) {

sum += a[i];

if(sum>tmpL) tmpL = sum;

}

sum = 0;

for (int i = mid+1; i <=R ; i++) {

sum += a[i];

if(sum>tmpR) tmpR = sum;

}

return Math.max(sumL,Math.max(sumR,tmpL+tmpR));

}

}

方法三、动态规划

c237fa5eed9873dc3d8ea5f3d72d796b.png

这种两端都是变化的问题是很难优化的,最好可以让一端固定,这样就会大大简化分析难度。于是将j暂时从原式子中提取出来,将剩下的命名为b[j]。

3835de51ce26ac9c0905cb433adf7a92.png

所以原问题就变成了这样。

93dfb006e15fd1de25dc82bf2fc626c3.png

根据b[j]的定义,b[j]是指以a[j]结尾的最大子段和。因此有如下公式:

b[j] = max{b[j - 1] + a[j] , a[j]}      1=

有了b[j],再对他取个极值,就可以得到原问题的解。

时间复杂度为O(n)

int maxSum(int[] a) {

int res = 0;

int b = 0;

for (int i = 0; i < a.length; i++) {

b = Math.max(b + a[i], a[i]);

if (b > res) res = b;

}

return res;

}

二、推广问题

最大子矩阵和问题

问题描述:给定一个m*n的整数矩阵A,试求矩阵A的一个子矩阵,使其各个元素之和最大。

问题分析:事实上,只需要将矩阵“压扁”就可以规约到最大子段和问题,具体来说就是将多行进行求和变为一行,这样就可以直接使用上述问题的解法。将多行“压缩”成一行有多种可行方案,需要遍历一下,花费O(m^2),最大子段和动态规划算法花费O(n),所以总的时间消耗是O(m^2*n)。

最大m子段和问题

问题描述:给定n个整数(可能有负数)组成的序列a1,a2,...,an,以及一个正整数m,要求确定该序列的m个不相交子段,使这m个子段的总和最大。

问题分析:设b(i, j)表示数组a的前j项中i个子段和的最大值,且第i个子段含a[j](1<=i<=m,i<=j<=n),则所求的最优值显然为max b(m, j),其中 m <= j <= n。

与最大子段和类似。计算b(i,j)的递归式子为:

b(i, j) = max{ b(i, j-1) + a[j] , max{ b(i - 1, t) + a[j] 其中t = i - 1 ~ j - 1} }

初始时,b(0, j) = 0; b(i, 0) = 0。

#include "stdafx.h"

#include

using namespace std;

int MaxSum(int m,int n,int *a);

int main() {

int a[] = {0,2,3,-7,6,4,-5};//数组脚标从1开始

for(int i=1; i<=6; i++) {

cout<

}

cout<

cout<

}

int MaxSum(int m,int n,int *a) {

if(n

return 0;

int **b = new int *[m+1];

for(int i=0; i<=m; i++) {

b[i] = new int[n+1];

}

for(int i=0; i<=m; i++) {

b[i][0] = 0;

}

for(int j=1;j<=n; j++) {

b[0][j] = 0;

}

//枚举子段数目,从1开始,迭代到m,递推出b[i][j]的值

for(int i=1; i<=m; i++) {

//n-m+i限制避免多余运算,当i=m时,j最大为n,可据此递推所有情形

for(int j=i; j<=n-m+i; j++) {

if(j>i) {

b[i][j] = b[i][j-1] + a[j];//代表a[j]同a[j-1]一起,都在最后一子段中

for(int k=i-1; k

if(b[i][j]

b[i][j] = b[i-1][k]+a[j];//代表最后一子段仅包含a[j]

}

}

else {

b[i][j] = b[i-1][j-1]+a[j];//当i=j时,每一项为一子段

}

}

}

int sum = 0;

for(int j=m; j<=n; j++) {

if(sum

sum = b[m][j];

}

}

return sum;

}

上述算法显然需要O(m*n^2)计算时间和O(m*n)。可以看一下具体矩阵是怎么填写的。

5f1a8d8feb66f1a83e47657a8b92fb6c.png

注意到上述算法中,计算b[i][j]时只用到了b的当前行的前一个数以及上一行的一个极值。因此我们可以定义两个数组,一个数组来保存当前行,一个数组来保存上一行的极值。并且使用数组来保存极值可以边生成当前行的数值边进行极值的判断并进行填充。

#include "stdafx.h"

#include

using namespace std;

int MaxSum(int m,int n,int *a);

int main() {

int a[] = {0,2,3,-7,6,4,-5};//数组脚标从1开始

for(int i=1; i<=6; i++) {

cout<


http://chatgpt.dhexx.cn/article/6xFLVrDV.shtml

相关文章

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

题目描述 给定一段长度为n的序列&#xff0c;我们需要找到其中一个连续的子段&#xff0c;使这个子段中各个元素加和最大&#xff0c;如果这个数组中全为负整数&#xff0c;我们就定义这个子段和为0. 题目分析 首先我们的目的是找一个局部的子段但加和是全局最大&#xff0c;…

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

什么是最大子段和&#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;得到长度更长的有序数组。当前合并好的有序数组为下…