一.问题描述
 
      给定长度为n的整数序列,a[1...n], 求[1,n]某个子区间[i , j]使得a[i]+…+a[j]和最大.或者求出最大的这个和.
      例如(-2,11,-4,13,-5,2)的最大子段和为20,所求子区间为[2,4].
      如果该序列的所有元素都是负整数时定义其最大子段和为0。
 
 二. 问题分析
      1、最大子段和问题的简单算法:
 
 
 
 
      2、最大子段和问题的分治法:
 
      求子区间及最大和,从结构上是非常适合分治法的,因为所有子区间[start, end]只可能有以下三种可能性:
      在[1, n/2]这个区域内
      在[n/2+1, n]这个区域内
      起点位于[1,n/2],终点位于[n/2+1,n]内
 int DAC(int * array, int left, int right)
 {
     if (left == right)
          ? array[left] : ;

     ;
     int leftSum = DAC(array, left, center);
     , right);

     ;
     ;
     for (int i=center;i>=left;--i)
     {
         temp += array[i];
         if (leftHalfMaxSum < temp)
             leftHalfMaxSum = temp;
     }
     temp = ;
     ;
     ;i<=right;++i)
     {
         temp += array[i];
         if (rightHalfMaxSum  < temp)
             rightHalfMaxSum = temp;
     }

     int max = leftSum > rightSum ? leftSum : rightSum;
     return max > leftHalfMaxSum + rightHalfMaxSum ? max : leftHalfMaxSum + rightHalfMaxSum;
 }

分治法的难点在于第三种情形的理解,这里应该抓住第三种情形的特点,也就是中间有两个定点,然后分别往两个方向扩张,以遍历所有属于第三种情形的子区间,求的最大的      一个,如果要求得具体的区间,稍微对上述代码做点修改即可. 分治法的计算时间复杂度为O(nlogn).

    3、最大子段和问题的动态规划算法:
      令b[j]表示以位置 j 为终点的所有子区间中和最大的一个
      子问题:如j为终点的最大子区间包含了位置j-1,则以j-1为终点的最大子区间必然包括在其中
      如果b[j-1] >0, 那么显然b[j] = b[j-1] + a[j],用之前最大的一个加上a[j]即可,因为a[j]必须包含
      如果b[j-1]<=0,那么b[j] = a[j]。
 
      对于这种子问题结构和最优化问题的证明,可以参考算法导论上的“剪切法”,即如果不包括子问题的最优解,把你假设的解粘帖上去,会得出子问题的最优化矛盾.证明如下:
      令a[x,y]表示a[x]+…+a[y] , y>=x
      假设以j为终点的最大子区间 [s, j] 包含了j-1这个位置,以j-1为终点的最大子区间[ r, j-1]并不包含其中
      即假设[r,j-1]不是[s,j]的子区间
      存在s使得a[s, j-1]+a[j]为以j为终点的最大子段和,这里的 r != s 
      由于[r, j -1]是最优解, 所以a[s,j-1]<a[r, j-1],所以a[s,j-1]+a[j]<a[r, j-1]+a[j]
      与[s,j]为最优解矛盾.
 int DP(int *a, int size)
 {
     int *b = new int[size];
     b[] = a[];
     ];
     ;i<size;++i)
     {
         ] > )
             b[i] = b[i-] + a[i];
         else
             b[i] = a[i];

         if(b[i]>max)
             max = b[i];
     }
     return max;
 }

测试代码:

 #include "stdafx.h"
 #include <stdlib.h>
 #include "DivideAndConquer.h"
 #include "DynamicProgramming.h"

 int _tmain(int argc, _TCHAR* argv[])
 {
     , , -, , -, -};
     //int result = DAC(array, 0, 5);
     );
     printf("%d", result);
     system("pause");
     ;
 }

转自:http://blog.csdn.net/jiyanfeng1/article/details/8058604

转载:最大子段和问题(Maximum Interval Sum)的更多相关文章

  1. leetcode&ndash;Binary Tree Maximum Path Sum

    1.题目说明 Given a binary tree, find the maximum path sum.   The path may start and end at any node in t ...

  2. [LeetCode] Binary Tree Maximum Path Sum 求二叉树的最大路径和

    Given a binary tree, find the maximum path sum. The path may start and end at any node in the tree. ...

  3. [leetcode]Binary Tree Maximum Path Sum

    Binary Tree Maximum Path Sum Given a binary tree, find the maximum path sum. The path may start and ...

  4. LeetCode(124) Binary Tree Maximum Path Sum

    题目 Given a binary tree, find the maximum path sum. For this problem, a path is defined as any sequen ...

  5. LeetCode124:Binary Tree Maximum Path Sum

    题目: Given a binary tree, find the maximum path sum. The path may start and end at any node in the tr ...

  6. leetcode 124. Binary Tree Maximum Path Sum

    Given a binary tree, find the maximum path sum. For this problem, a path is defined as any sequence ...

  7. [lintcode] Binary Tree Maximum Path Sum II

    Given a binary tree, find the maximum path sum from root. The path may end at any node in the tree a ...

  8. 【leetcode】Binary Tree Maximum Path Sum

    Binary Tree Maximum Path Sum Given a binary tree, find the maximum path sum. The path may start and ...

  9. 【leetcode】Binary Tree Maximum Path Sum (medium)

    Given a binary tree, find the maximum path sum. The path may start and end at any node in the tree. ...

随机推荐

  1. Android和JavaScript相互调用的方法

    转载地址:http://www.jb51.net/article/77206.htm 这篇文章主要介绍了Android和JavaScript相互调用的方法,实例分析了Android的WebView执行 ...

  2. asp 使用TreeView控件

    这段代码为了使用 TreeNodeCheckChanged 事件,会有回刷新的效果: 不喜欢的可查看改进版,利用js控制选择操作,无界面刷新, “http://www.cnblogs.com/GoCi ...

  3. sql月份销售统计

    1.SELECT   MONTH(SellTime) as selltime,SUM(TotalPrice) as total FROM  Sell WHERE  YEAR(SellTime)=CON ...

  4. Razor练习4

    今天练习Razor的逻辑处理.一般会使用下面 1. if 2. else 3. else if 4. switch 下面演示中,Insus.NET分别演示1,2, 4:xxx.cshtml代码如下: ...

  5. 更改MySql表和字段区分大小写

    数据库和表名在 Windows 中是大小写不敏感的 ,而在大多数类型的 Unix 系统中是大小写敏感的Windows 版的 MySQL 默认继承 os 的大小写习惯,即使 SQL中有区分,在导入的时候 ...

  6. java基础-在dos控制台编写简易 的java程序

    第一步:在文件夹中修改隐藏的文件扩展名,让其文件的扩展名全部显示:————win7系统在文件的组织下方的文件夹和搜索选项,选择查看,将里面的隐藏选项取消: 第二步:在文件夹中新建一个text文件,将其 ...

  7. Linux常用命令_(文件操作)

    对文件的操作主要有以下命令: touch.cp.rm.mv.ln.mkdir.rmdir

  8. cxf动态调用wsdl的一个冲突以及解决

    cxf发布服务,调用服务的博客很多,这里也就简单贴一下代代码. 环境如下:spring+cxf (maven环境) <cxf.version>2.7.11</cxf.version& ...

  9. java线程中断和终止线程运行

    ava中启动一个线程很容易,通常情况下我们都是等到任务运行结束后让线程自行停止.但有时需要在任务正在运行时取消他们,使得线程快速结束.对此Java并没有提供任何机制.但是我们可以通过Java提供的线程 ...

  10. C#拉姆达(=&gt;)表达式

    前言: 之前小猪曾经分享过自己对C#委托的一点理解 其实在使用委托的过程中我们会大量的使用拉姆达(=>)表达式 介绍: "Lambda表达式"是一个匿名函数,是一种高效的类似 ...