分治策略(求最大子数组和)

思想:

  1. 分治:将原问题分解为许多小问题,使小问题的形式与原问题相同,只是问题的规模更小。
  2. 求解:递归地求解出子问题,当子问题的规模足够小时,停止递归,解决问题。
  3. 合并:将子问题的解合并起来,组合成原问题的解。

例 求最大子数组和(leetcode 53T)

给你一个整数数组 nums ,请你找出一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。

子数组 是数组中的一个连续部分。
示例 1:

输入:nums = [-2,1,-3,4,-1,2,1,-5,4] 输出:6 解释:连续子数组 [4,-1,2,1] 的和最大,为 6 。
示例 2:

输入:nums = [1] 输出:1
示例 3:

输入:nums = [5,4,-1,7,8] 输出:23


求解nums[low,high]的最大子数组,根据分治策略,将原问题分解成同等形式的小问题,则可以以中间(mid)为标准进行分割,分成nums[low,mid] 和num [mid+1,high]两个子数组,然后对每个子问题以此类推,直至子问题不可再分。此种情况下,nums[low,high]中的最大连续子数组nums[i,j]必然为以下三种情况之一:

  • 完全在 nums[low,mid]中
  • 完全在nums[mid+1,high]中
  • 跨越两个子数组,low < i < mid < j < high。

cpp 复制代码
class Solution {
public:
int maxCrossSum(vector<int>& nums,int low,int mid,int high)
{
    int leftSum = -2147483648;
    int sum = 0;
    for(int i = mid;i >=low;i--){
        sum += nums[i];
        if(sum > leftSum){
            leftSum = sum;
        }
    }
    int rightSum = -2147483648;
    sum = 0;
    for(int i = mid + 1;i <=high;i++){
        sum += nums[i];
        if(sum > rightSum){
            rightSum = sum;
        }
    }
    return leftSum + rightSum;
}

int maxLRSum(vector<int>& nums,int low,int high)
{
    if(low == high){
        return nums[low];
    }
    else{
        int mid = (low + high)/2;
        int leftArr = 0;
        leftArr = maxLRSum(nums,low,mid);
        int rightArr = 0;
        rightArr = maxLRSum(nums,mid + 1,high);
        int midArr = 0;
        midArr = maxCrossSum(nums,low,mid,high);
        if(leftArr >= rightArr && leftArr >= midArr){
            return  leftArr;
        }
        if(rightArr>= leftArr && rightArr >= midArr){
            return rightArr;
        }
        if(midArr>= leftArr&& midArr >= rightArr){
            return midArr;
        }
    }
    return 0;
}
int maxSubArray(vector<int>& nums) {
    int low = 0;
    int high = nums.size()-1;
    int sum = 0;
    sum = maxLRSum(nums,low,high);
    return sum;
	}
};
相关推荐
椰萝Yerosius13 分钟前
[题解]2024CCPC郑州站——Z-order Curve
c++·算法
滨HI03 小时前
C++ opencv简化轮廓
开发语言·c++·opencv
学习路上_write3 小时前
FREERTOS_互斥量_创建和使用
c语言·开发语言·c++·stm32·单片机·嵌入式硬件
闻缺陷则喜何志丹5 小时前
【SOSDP模板 容斥原理 逆向思考】3757. 有效子序列的数量|分数未知
c++·算法·力扣·容斥原理·sosdp·逆向思考
BestOrNothing_20155 小时前
一篇搞懂 C++ 重载:函数重载 + 运算符重载,从入门到会用(含 ++、<<、== 实战)
c++·函数重载·运算符重载·operator·前置后置++·重载与重写
2501_941144425 小时前
Python + C++ 异构微服务设计与优化
c++·python·微服务
程序猿编码5 小时前
PRINCE算法的密码生成器:原理与设计思路(C/C++代码实现)
c语言·网络·c++·算法·安全·prince
CoderYanger6 小时前
优选算法-字符串:63.二进制求和
java·开发语言·算法·leetcode·职场和发展·1024程序员节
charlie1145141916 小时前
深入理解C/C++的编译链接技术6——A2:动态库设计基础之ABI设计接口
c语言·开发语言·c++·学习·动态库·函数
Cx330❀6 小时前
C++ STL set 完全指南:从基础用法到实战技巧
开发语言·数据结构·c++·算法·leetcode·面试