相关介绍:

 求取数组中最大连续子序列和问题,是一个较为“古老”的一个问题。该问题的描述为,给定一个整型数组(当然浮点型也是可以的啦),求取其下标连续的子序列,且其和为该数组的所有子序列和中值为最大的。例如数组A={1, 3, -2, 4, -5},则最大连续子序列和为6,即1+3+(-2)+ 4 = 6。解决该问题的算法有四种,根据其时间复杂度的高低,下面分别为这四种算法做介绍。

第一种:时间复杂度为O(N^3)

 该算法也是最容易想到的,很直观的算法,其算法的思路为,穷举数组中以某个元素为起始元素,以另一个元素为终止元素的元素子序列的和,并记录找出的子序列和最大的那个值

示例代码如下:

/**
* 用于演示O(N^3)的算法
* @param array 用于寻找最大连续子序列和的数组
* @return 最大连续子序列和的值
*/
public int maxSubN3(int[] array)
{
int result=0;
int length=array.length;
//用于控制子序列的起始下标
for(int i=0;i<length;i++)
{
//用于控制子序列的终止下标
for(int j=i;j<length;j++)
{
//用于计算起始下标到终止下标之间的元素的和
int sum=0;
for(int k=i;k<=j;k++)
{
sum+=array[k];
}
result=Math.max(sum, result);
}
}
return result;
}

第二种:时间复杂度为O(N^2)

 该算法通过对上面算法的实现方式进行改进,从而降低了时间复杂度。该算法的思路同上一个的大致相同,但是注意到,每次将终止元素的下标往后移动一位的时候,其子序列和又重新计算了一遍。为此,可以通过保留其上一次从起始元素到终止元素的之间的子序列和的计算结果,来减低时间复杂度。

示例代码如下:

/**
* 用于演示O(N^2)的算法
* @param array 用于寻找最大连续子序列和的数组
* @return 最大连续子序列和的值
*/
public int maxSubN2(int[] array)
{
int length=array.length;
int result=0;
for(int i=0;i<length;i++)
{
int sum=0;
for(int j=i;j<length;j++)
{
sum+=array[j];
result=Math.max(result, sum);
}
}
return result;
}

第三种:时间复杂度为O(NlogN)

 采用分治的思想,将数组划分为两半,最大连续子序列和出现的地方只可能有三种,要么在数组的左半边,要么在数组的右半边,要么横跨了数组的左右半边,同时其必定包含数组左半边的右边界以及数组右半边的左边界

/**
* 用于演示O(NlogN)的算法
* @param array 用于寻找最长子序列的数组
* @return
*/
public int maxSubNlogN(int[] array)
{
return maxSubNlogN(array,0,array.length-1);
}
/**
* 用于递归的那个方法,low和high为数组的左右边界
*/
private int maxSubNlogN(int[] array,int low,int high)
{
//当只剩下一个数组元素的时候,将其直接进行返回
if(low==high)
return Math.max(array[low], 0);
//找到数组中间的分界点
int mid=low+(high-low)/2;
//用于寻找到横跨数组左右半边的那个最长子序列
int lMax=0,rMax=0;
int lSum=0,rSum=0;
//找到包含右边界的左半边的最长子序列
for(int i=mid;i>=low;i--)
{
lSum+=array[i];
lMax=Math.max(lSum, lMax);
}
//找到包含左边界的右半边的最长子序列
for(int i=mid+1;i<high;i++)
{
rSum+=array[i];
rMax=Math.max(rSum, rMax);
}
//返回三种情况中子序列和最大的那个值
return Math.max(maxSubNlogN(array,mid+1,high),Math.max(rMax+lMax, maxSubNlogN(array,low,mid)));
}

第三种:时间复杂度为O(N)

 由分析可知,数组的最大连续子序列中的包含最大连续子序列的开头元素或者末尾元素的任意一个子序列里的元素值之和均为正数。为此,可以在仅需遍历一遍数组的过程中,计算出其最大连续子序列和

示例代码如下:

/**
* 用于演示O(N)的算法
* @param array 用于寻找最长子序列的数组
* @return 最长子序列的值
*/
public int maxSubN(int[] array)
{
int sum=0;
int result=0;
for(int i=0;i<array.length;i++)
{
sum+=array[i];
if(sum>result)
result=sum;
if(sum<0)
sum=0;
}
return result;
}

代码汇总如下:

package other;
/**
* 该类用于演示寻找最长子序列的几种算法
* @author 学徒
*
*/
public class MaxSubLined
{
public static void main(String[] args)
{
int[] array={-2,11,-4,13,-5,2,-5,-3,12,-9};
MaxSubLined max=new MaxSubLined();
System.out.println(max.maxSubN3(array));
System.out.println(max.maxSubN2(array));
System.out.println(max.maxSubNlogN(array));
System.out.println(max.maxSubN(array));
}
/**
* 用于演示O(N^3)的算法
* 算法的思路:
* 穷举数组中以某个元素为起始的,数组中以该元素为起始元素的所有子序列的可能
* 并找出其中子序列和最大的那个值
* @param array 用于寻找最长子序列的数组
* @return 最长子序列的值
*/
public int maxSubN3(int[] array)
{
int result=0;
int length=array.length;
//用于控制子序列的起始下标
for(int i=0;i<length;i++)
{
//用于控制子序列的终止下标
for(int j=i;j<length;j++)
{
//用于计算起始下标到终止下标之间的元素的和
int sum=0;
for(int k=i;k<=j;k++)
{
sum+=array[k];
}
result=Math.max(sum, result);
}
}
return result;
}
/**
* 用于演示O(N^2)的算法
* 算法的思路:
* 穷举数组中以某个元素为起始的,数组中以该元素为起始元素的所有子序列的可能
* 并找出其中子序列和最大的那个值
* @param array 用于寻找最长子序列的数组
* @return 最长子序列的值
*/
public int maxSubN2(int[] array)
{
int length=array.length;
int result=0;
for(int i=0;i<length;i++)
{
int sum=0;
for(int j=i;j<length;j++)
{
sum+=array[j];
result=Math.max(result, sum);
}
}
return result;
}
/**
* 用于演示O(NlogN)的算法
* 算法的思路:
* 采用分治的思想,将数组划分为两半,最长子序列出现的可能只有三种
* 要么在数组的左半边,要么在数组的右半边,要么横跨了数组的左右半边,同时其必定包含
* 数组左半边的右边界以及数组右半边的左边界
* @param array 用于寻找最长子序列的数组
* @return
*/
public int maxSubNlogN(int[] array)
{
return maxSubNlogN(array,0,array.length-1);
}
/**
* 用于递归的那个方法,low和high为数组的左右边界
*/
private int maxSubNlogN(int[] array,int low,int high)
{
//当只剩下一个数组元素的时候,将其直接进行返回
if(low==high)
return Math.max(array[low], 0);
//找到数组中间的分界点
int mid=low+(high-low)/2;
//用于寻找到横跨数组左右半边的那个最长子序列
int lMax=0,rMax=0;
int lSum=0,rSum=0;
//找到包含右边界的左半边的最长子序列
for(int i=mid;i>=low;i--)
{
lSum+=array[i];
lMax=Math.max(lSum, lMax);
}
//找到包含左边界的右半边的最长子序列
for(int i=mid+1;i<high;i++)
{
rSum+=array[i];
rMax=Math.max(rSum, rMax);
}
//返回三种情况中子序列和最大的那个值
return Math.max(maxSubNlogN(array,mid+1,high),Math.max(rMax+lMax, maxSubNlogN(array,low,mid)));
}
/**
* 用于演示O(N)的算法
* 算法的思路:
* 因为最长子序列中的任意一个子序列里的元素值总和均为正数
* 为此,可以在遍历数组的过程中,计算出其最长子序列
* @param array 用于寻找最长子序列的数组
* @return 最长子序列的值
*/
public int maxSubN(int[] array)
{
int sum=0;
int result=0;
for(int i=0;i<array.length;i++)
{
sum+=array[i];
if(sum>result)
result=sum;
if(sum<0)
sum=0;
}
return result;
}
}

回到目录|·(工)·)

K:求取数组中最大连续子序列和的四个算法的更多相关文章

  1. 求一个数组中最小的K个数

    方法1:先对数组进行排序,然后遍历前K个数,此时时间复杂度为O(nlgn); 方法2:维护一个容量为K的最大堆(<算法导论>第6章),然后从第K+1个元素开始遍历,和堆中的最大元素比较,如 ...

  2. python实现给定K个字符数组,从这k个字符数组中任意取一个字符串,按顺序拼接,列出所有可能的字符串组合结果!

    题目描述:给定K个字符数组,从这k个字符数组中任意取一个字符串,按顺序拼接,列出所有可能的字符串组合结果! 样例: input:[["a","b"," ...

  3. 转载——JavaScript学习笔记:取数组中最大值和最小值

    转载自:http://www.w3cplus.com/javascript/calculate-the-max-min-value-from-an-array.html. 取数组中最大值 可以先把思路 ...

  4. JavaScript学习:取数组中最大值和最小值

    在实际业务中有的时候要取出数组中的最大值或最小值.但在数组中并没有提供arr.max()和arr.min()这样的方法.那么是不是可以通过别的方式实现类似这样的方法呢?那么今天我们就来整理取出数组中最 ...

  5. 求出数组中所有数字的和&&弹出层效果

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  6. 黑马基础阶段测试题:定义一个int类型的数组,数组中元素为{5,7,3,9,4}。求出数组中的最小值,并判断最小值是否为偶数,如果是偶数则输出“最小值为偶数”,如果不是偶数则输出“最小值为奇数”。打印如下:

    package com.swift; import java.util.Arrays; public class ArrayTest { public static void main(String[ ...

  7. 《程序员代码面试指南》第七章 位运算 在其他数都出现k 次的数组中找到只出现一次的数

    题目 在其他数都出现k 次的数组中找到只出现一次的数 java 代码 package com.lizhouwei.chapter7; /** * @Description: 在其他数都出现k 次的数组 ...

  8. 如何求出数组中最小(或者最大)的k个数(least k问题)

    输入n个整数,如何求出其中最小的k个数? 解法1. 当然最直观的思路是将数组排序,然后就可以找出其中最小的k个数了,时间复杂度以快速排序为例,是O(nlogn): 解法2. 借助划分(Partitio ...

  9. [LeetCode] Split Array into Consecutive Subsequences 将数组分割成连续子序列

    You are given an integer array sorted in ascending order (may contain duplicates), you need to split ...

随机推荐

  1. 爬虫3:requests库

      一个简单易用的http库,多用于第一步,爬取网站源码   简单例子 import requests   response = requests.get('https://www.baidu.com ...

  2. Nginx文件上传下载实现与文件管理

    1.Nginx 上传 Nginx 依赖包下载 # wget http://www.nginx.org/download/nginx-1.2.2.tar.gzinx # wget http://www. ...

  3. Tomcat 基础优化

    作者:北京运维 本文档是身边一些朋友.技术大佬之前分享的一些笔记,记录了 Tomcat 优化方法,笔记较多而且比较杂乱,经过整理.分类我个人觉得大致可以从以下几个方面优化 Tomcat: Tomcat ...

  4. Linux文件索引节点相关概念

    一.  概念 1.  inode(index node)表中包含文件系统所有文件列表 一个节点 (索引节点)是在一个表项,包含有关文件的信息( 元数据 ),包括: 文件类型,权限,UID,GID 链接 ...

  5. loj 6433 「PKUSC2018」最大前缀和 题解【DP】【枚举】【二进制】【排列组合】

    这是个什么集合DP啊- 想过枚举断点但是不会处理接下来的问题了- 我好菜啊 题目描述 小 C 是一个算法竞赛爱好者,有一天小 C 遇到了一个非常难的问题:求一个序列的最大子段和. 但是小 C 并不会做 ...

  6. win10 下springcloud打包docker镜像部署。

    1:建一个最简单的springcloud应用. 2:在根目录下新建dockerfile,文件如下: FROM openjdk:8-jdk-alpine VOLUME /tmp ARG JAR_FILE ...

  7. redis 持久化之 rdb 快照持久化

    解释1: 虽然redis是单进程,但是它有一个单独的子进程进行rdb操作,为了保证的数据的一致性,当进行rdb操作失败的时候,主进程就停止写入 所以才有了stop-write-on-bgsave-er ...

  8. 执行Hive时出现org.apache.hadoop.util.RunJar.main(RunJar.java:136) Caused by: java.lang.NumberFormatException: For input string: "1s"错误的解决办法(图文详解)

    不多说,直接上干货 问题详情 [kfk@bigdata-pro01 apache-hive--bin]$ bin/hive Logging initialized -bin/conf/hive-log ...

  9. HTTPS的误解(二)

    大家好,我们接着HTTPS的误解(一)接着讲,经常有人会说更换或转移服务器时要购买新证书,服务器SSL证书价格很贵,易维信(EVTrust)给大家澄清了这些容易产生误解的地方,详细见下面文章. 误解四 ...

  10. 我与ARM的那些事儿1初识ARM

    最近一直在研究ARM,说到ARM,我们首先想到了是三星.高通等公司,这些公司都制造CPU的,其实ARM也是一家公司,只不过它是提供最核心的逻辑电路,而且它的赚钱方式是与其他公司进行双赢的!你卖出多少芯 ...