算法设计--在数组中找求和最大的连续子串

问题:输入具有n个整数的向量arr,输出向量的任意连续子向量和的最大值

特殊情况(1、当向量都为正数时,为整个向量

     2、当向量都为负数时,为0,即空子串

    )

1、O(n2)的算法 (循环对所有情况进行遍历)

 1 #include <stdio.h>
 2 #define max(a,b) ((a>b)?a:b)
 3 #define max3(a,b,c) ((a>b)?((a>c)?a:c):((b>c)?b:c))
 4
 5 int find1(int arr[], int n){
 6     int i,j,sum,maxsofar;
 7     maxsofar = 0;
 8
 9     for(i=0; i<n; i++){
10         sum = 0;
11         for(j=i; j<n; j++){
12             sum += arr[j];
13             maxsofar = max(sum, maxsofar);
14         }
15     }
16     return maxsofar;
17 }

其中有个小细节就是 注意sum(i, j-1) 和 sum(i, j)的关系,不要每次在求和的时候从头(i的位置)开始,那样会使复杂度变为O(n3)



2、O(nlogn)算法

基于分治原理的算法:首先将n的原问题划分为大小基本相等的两个子问题,我们分别称为a和b子问题,可以递归找出a和b问题的最大子向量,称为maxa 和 maxb。

但他们两个之间的最大值不一定使我们求得n问题的最优解,还有一种可能是跨越a和b的边界,我们称之为c,c情况的最优解为maxc。

那么问题变成了如何求解maxc?

我们可以发现,maxc中在a的部分为a中包括a的右边界的最大值,maxc中在b的部分为b中包括b的左边界的最大值,因此可以在O(N)的时间内算出maxc

因此得到T(N) = 2T(N/2) + O(N)

推导得到T(N) = O(nlogn)

 1 int find2(int arr[], int s_p, int e_p){
 2     int m, sum, i, maxsofar, lmaxsofar, rmaxsofar;
 3     maxsofar = 0;
 4
 5     if(s_p == e_p){
 6         return maxsofar;
 7     }
 8     else if(s_p == e_p){
 9         return max(arr[s_p],0);
10     }
11     else{
12         m = (s_p + e_p) / 2;
13
14         lmaxsofar = 0;
15         sum = 0;
16         for(i=m; i>=s_p; i--){
17             sum += arr[i];
18             lmaxsofar = max(sum, lmaxsofar);
19         }
20
21         rmaxsofar = 0;
22         sum = 0;
23         for(i=m+1; i<=e_p; i++){
24             sum += arr[i];
25             rmaxsofar = max(sum, rmaxsofar);
26         }
27
28         return max3(lmaxsofar+rmaxsofar,find2(arr, s_p, m), find2(arr, m+1, e_p));
29     }
30 }


3、O(n)算法

先上代码,代码非常简短,理解起来比较困难,但是执行效率非常高

 1 int find3(int arr[], int n){
 2     int i,maxsofar,maxendinghere;
 3     maxsofar = 0;
 4     maxendinghere = 0;
 5
 6     for(i=0; i<n; i++){
 7         maxendinghere = max(maxendinghere + arr[i], 0);
 8         maxsofar = max(maxsofar, maxendinghere);
 9     }
10
11     return maxsofar;
12 }

假设我们已经解决了x[0,n-1]的问题,利用分治算法的原理:前i个元素中,最大总和子数组要么在前i-1个元素中,要么其结束位置在i处。

分析其结束为止在i处的情况,那么子向量中除去i处的元素组成的子向量一定是x[0,i-1]中结束位置为i-1的最大子向量。

看代码中的关键变量为maxendinghere:在循环语句的第一个赋值语句之前,maxendinghere是结束位置为i-1的最大子向量的和;赋值语句将其修改为结束位置为i的最大子向量的和。若加上x[i]后结果依然为正值,则结束位置在i的最大子向量值就为maxendinghere+x[i],如果为负值,则重置为0。

原文地址:https://www.cnblogs.com/lwyeah/p/8584122.html

时间: 2024-07-30 14:07:44

算法设计--在数组中找求和最大的连续子串的相关文章

C语言 选择排序算法原理和实现 从数组中 找出最小的元素然后交换位置

#include <stdio.h> int main(void) { /* 选择排序算法 原理:从数组中 找出最小的元素然后交换位置: */ int a[10] = {9,5,10,7,2,3,1,6,8,4}; int i=0,j=0; int n = sizeof(a)/4; //外循环n-1轮 for(i=0;i<n-1;i++){ int pos = i;//始终指向最小的位置 for(j=i+1;j<n;j++){ if(a[j]<a[pos]){ pos = j

算法题:找出一个数组中相加值最大的连续序列元素

package arithmetic; /** * @author SHI * 求一个数组中相加值最大的连续序列元素 */ public class MaxSequence { public static void main(String[] args) { int[] a=new int[]{-2,9,-3,4,-6,7,-6,4}; findBigSequence(a); } /** * 思想: (1)计算出该数组的所有元素和,假设该值为最大 * (2)从数组下标1到a.length-1依次

在数组中找几个数的和等于某个数[LeetCode]

首先明确一点,这个方面的问题设计到的知识点是数组的查找的问题.对于类似的这样的查找操作的具体办法就是三种解决方法: 1.暴力算法,多个for循环,很高的时间复杂度 2.先排序,然后左右夹逼,但是这样会破坏原始数组的下表 3.利用Hash表,直接定位元素,很少的时间复杂度 TwoSums 先来看看最简单的,在一个数组中找两个数的和等于某个数. 这个题目最简简单的方法就是暴力法,所需的时间复杂度是O(n2),但是这是不允许的,所以一个O(n)的方法就是利用Hash表存储数据,这样能够把查找的时间降低

[经典算法题]寻找数组中第K大的数的方法总结

[经典算法题]寻找数组中第K大的数的方法总结 责任编辑:admin 日期:2012-11-26 字体:[大 中 小] 打印复制链接我要评论 今天看算法分析是,看到一个这样的问题,就是在一堆数据中查找到第k个大的值. 名称是:设计一组N个数,确定其中第k个最大值,这是一个选择问题,当然,解决这个问题的方法很多,本人在网上搜索了一番,查找到以下的方式,决定很好,推荐给大家. 所谓"第(前)k大数问题"指的是在长度为n(n>=k)的乱序数组中S找出从大到小顺序的第(前)k个数的问题.

(算法)求数组中出现频率最高的数

不准备实现算法先,根据21题和前辈的经验,这道题的真正考核点不在于解决这个问题,而在于拿到这个问题以后题的问题. 正常的一个做法,一次扫描然后用HASHMAP进行一个统计,然后再扫描一次HASHMAP获得频率最高的数.时间是O(N)空间也是O(N). 还有一种是做排序,然后扫描一次根据下标计算可以得到频率最高的数(可以避免空间消耗?). (不知道是否还有别的做法?) 据说我们应该先问这个数组是否已经排序?(想得美) 然后是否已经知道这个数的大概出现频率(比如说超过一半) 是否可以用额外空间?是否

脑洞题目 - 改自从一组无序数组中找不存在的最小正整数

无聊想的题目,但题目创意改自从一组无序数组中找不存在的最小正整数.并不难,但也有陷阱,比考虑0和负数... /** * 从无序数组中找不存在的最小正整数 * 我的要求:比如:{3,4,6,9,20}中 最小的不存在的正整数为2 */ #include <stdio.h> int main() { int arr[] = {-5, -10, 42, 29, 18, -3, 8, 20, -1}; int i, j, temp; for(i = 1; i < sizeof(arr)/size

bestcoder#43 1002 在数组中找两个数的和取模的最大值 二分

bestcoder#43 1002 在数组中找两个数的和取模的最大值  二分 pog loves szh II Accepts: 97 Submissions: 834 Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) Problem Description Pog and Szh are playing games. There is a sequence with n number

有序数组中找中位数

问题描述 给定两个有序数组,返回这两个数组的中位数.如果中位数有两个,则返回它们的平均值. e.g. [1, 3, 5]和[2, 4, 6]的中位数是3.5 解决思路 如果两个数组的长度之和为奇数,则中位数有一个:否则中位数为其中两个的平均值. 从两个数组中找第k个数,可以使用递归的思路. 程序 首先,写出在有序数组a和b中找到第k大的程序: 1. 利用归并排序中的merge数组方法,时间复杂度为O(k). public int findKthNaive(int[] a, int[] b, in

C语言:对传入sp的字符进行统计,三组两个相连字母“ea”&quot;ou&quot;&quot;iu&quot;出现的次数,并将统计结果存入ct所指的数组中。-在数组中找出最小值,并与第一个元素交换位置。

//对传入sp的字符进行统计,三组两个相连字母“ea”"ou""iu"出现的次数,并将统计结果存入ct所指的数组中. 1 #include <stdio.h> 2 #include <string.h> 3 #pragma warning (disable:4996) 4 void fun(char*sp ,int *ct) 5 { 6 int a=0, b=0, c=0; 7 while (*sp != '\0') 8 { 9 if (*s