求最大子数组之和的方法解析(2种可选)

 更新时间:2016年12月23日 16:36:59   作者:一个弱者想变强  
本文主要对求最大子数组之和的方法进行详细解析,列了两种方法供大家选择借鉴,需要的朋友一起来看下吧

问题描述:一个有n个元素的数组,这n个元素可以是正数也可以是负数,求最大子数组的和。

方法1:蛮力法

思路:最简单也是最容易想到的方法就是找出所有子数组,然后求所有子数组的和,在所有子数组的和中取最大值。

/**
  * 方法1(蛮力法):两次循环求最大子数组之和
  */
 public static int maxSubArray1(int[] a){
  int i,j;
  int ThisSum=0;
  int MaxSum=0;
  for (i = 0; i < a.length; i++) {
   ThisSum=a[i];
   for(j=i+1;j<a.length;j++){
    ThisSum+=a[j];
    if(ThisSum>MaxSum){
     MaxSum=ThisSum;
    }
   }
  }
  return MaxSum;
 }

方法2:优化的动态规划

思路:首先可以根据数组的最后一个元素a[n-1]与最大子数组的关系分为以下三种情况:

1) 最大子数组包含a[n-1],即以a[n-1]结尾。

2) a[n-1]单独构成最大子数组。

3) 最大子数组不包含a[n-1],那么求a[1,...,n-1]的最大子数组可以转换为求a[1,...,n-2]的最大子数组。

通过上述分析可以得出如下结论:假设已经计算出(a[0],...a[i-1])最大的一段数组和为All[i-1],同时也计算出(a[0],...a[i-1])中包含a[i-1]的最大的一段数组和为End[i-1],

则可以得出如下关系:All[i-1]=max{a[i-1],End[i-1],All[i-1]}。利用这个公式和动态规划的思想解决问题。(代码中还解决了起始位置,终止位置的问题)

/**
  * 方法2:优化的动态规划方法
  * nEnd就是通过“数组依次相加加到a[i],然后与a[i]做比较”得来的,保存较大的。因为如果前面的数加到a[i]
  * 还没有a[i]本身大,那么前面的数也就对最大子数组和没有贡献。厉害
  * nAll就是记录一下之前的新得到的nEnd和自身之前谁更大
  */
 public static int max(int m,int n){
  return m>n?m:n;
 }
 public static int maxSubArray2(int[] a){
  int nAll=a[0];//有n个数字数组的最大子数组之和
  int nEnd=a[0];//有n个数字数组包含最后一个元素的子数组的最大和
  for (int i = 1; i < a.length; i++) {
   nEnd=max(nEnd+a[i],a[i]);
   nAll=max(nEnd, nAll);
  }
  return nAll;
 }
 private static int begin=0;
 private static int end=0;
 /**
  * 求出最大子数组的开始begin,结尾end,以及整个子数组
  */
 public static int maxSubArray3(int[] a){
  int maxSum=Integer.MIN_VALUE;
  int nSum=0;
  int nStart=0;
  for (int i = 0; i < a.length; i++) {
   if(nSum<0){
    nSum=a[i];
    nStart=i;
   }
   else{
    nSum+=a[i];
   }
   if(nSum>maxSum){
    maxSum=nSum;
    begin=nStart;
    end=i;
   }
  }
  return maxSum;
 }

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持脚本之家!

相关文章

  • JSON反序列化Long变Integer或Double的问题及解决

    JSON反序列化Long变Integer或Double的问题及解决

    这篇文章主要介绍了JSON反序列化Long变Integer或Double的问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-01-01
  • Java流程控制之循环结构for,增强for循环

    Java流程控制之循环结构for,增强for循环

    这篇文章主要介绍了Java流程控制之循环结构for,增强for循环,for循环是编程语言中一种循环语句,而循环语句由循环体及循环的判定条件两部分组成,其表达式为:for(单次表达式;条件表达式;末尾循环体){中间循环体;},下面我们俩看看文章内容的详细介绍
    2021-12-12
  • javaweb如何使用华为云短信通知公共类调用

    javaweb如何使用华为云短信通知公共类调用

    这篇文章主要介绍了javaweb使用华为云短信通知公共类调用的操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-06-06
  • Java中的键盘事件处理及监听机制解析

    Java中的键盘事件处理及监听机制解析

    这篇文章主要介绍了Java中的键盘事件处理及监听机制解析,Java事件处理采用了委派事件模型,在这个模型中,当事件发生时,产生事件的对象将事件信息传递给事件的监听者进行处理,在Java中,事件源是产生事件的对象,比如窗口、按钮等,需要的朋友可以参考下
    2023-10-10
  • JAVA中的for循环几种使用方法讲解

    JAVA中的for循环几种使用方法讲解

    这篇文章主要介绍了JAVA中的for循环几种使用方法讲解,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-09-09
  • java中对list分页并显示数据到页面实例代码

    java中对list分页并显示数据到页面实例代码

    这篇文章主要介绍了java中对list分页并显示数据到页面实例代码,分享了相关代码示例,小编觉得还是挺不错的,具有一定借鉴价值,需要的朋友可以参考下
    2018-02-02
  • Mybatis实现增删改查

    Mybatis实现增删改查

    这篇文章主要介绍了Mybatis实现增删改查,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-01-01
  • idea创建springboot项目,java版本只能选择17和21的解决方案

    idea创建springboot项目,java版本只能选择17和21的解决方案

    这篇文章主要介绍了idea创建springboot项目,java版本只能选择17和21的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2025-04-04
  • 如何使用Reactor完成类似Flink的操作

    如何使用Reactor完成类似Flink的操作

    这篇文章主要介绍了如何使用Reactor完成类似Flink的操作,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下
    2021-03-03
  • 常见的java面试题

    常见的java面试题

    这篇文章主要为大家详细介绍了常见的java面试题,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-11-11

最新评论