C++算法前缀和的应用:分割数组的最大值的原理、源码及测试用例

分割数组的最大值

相关知识点

C++算法:前缀和、前缀乘积、前缀异或的原理、源码及测试用例:付视频课程

二分 过些天整理基础知识

题目

给定一个非负整数数组 nums 和一个整数 m ,你需要将这个数组分成 m 个非空的连续子数组。

设计一个算法使得这 m 个子数组各自和的最大值最小。

示例 1:

输入:nums = 7,2,5,10,8, m = 2

输出:18

解释:

一共有四种方法将 nums 分割为 2 个子数组。

其中最好的方式是将其分为 7,2,510,8

因为此时这两个子数组各自的和的最大值为18,在所有情况中最小。

示例 2:

输入:nums = 1,2,3,4,5, m = 2

输出:9

示例 3:

输入:nums = 1,4,4, m = 3

输出:4

提示:

1 <= nums.length <= 1000

0 <= numsi <= 10^6

1 <= m <= min(50, nums.length)

解法一:暴力解法

时间复杂度O(nnm),n是nums的长度。vMaxSum共有m*n种状态,求每种状态需的时间复杂度是O(n)。vPreSum记录前缀和,vMaxSumij 记录将nums0,j分成i个子数组的最大和。j'取值范围0,j),vMaxSum\[ij就是所有max(vMaxSumi-1j',vPreSumj+1 - vPreSumj')的最小值。这个时间复杂度在通过和不通过的边缘。

解法二:滑动窗口

假定j的j1是x,则当j增加时,x不变或增加。 当j++,vMaxSumi-1j'不变,vPreSumj+1 - vPreSumj' 增加。下面用因果表来证明。令L(j,x)= vMaxSumi-1x R(j,x) = vPreSumj+1 - vPreSumx |。

如果L(j,x)<= R(j,x)。x减少后,左式减少或不变,右式增加或不变。i++后,右式变大或不变。所以x减少只会让右式变大或不变。而右式显然大于左式,所以减少左式不会减少最大值。

规章编号 证明
假设一 合适的j1就是x
假设二 L(j,x)> R(j,x)
推论一 假设一 假设二 x--后,L变小,R变大。如果L(j,x-1) >= R(j,x-1),结合假设二,x-1比x更合适。与假设一矛盾。 L(j,x-1) < R(j,x-1)]
推论二 对于j+1,取x最大和为L(j,x)或R(j+1,x);取x-1,最大和为R(j+1,x-1)

代码

class Solution {

public:

int splitArray(vector& nums, int k) {

m_c = nums.size();

vector vPreSum(1);

for (const auto& n : nums)

{

vPreSum.emplace_back(n + vPreSum.back());

}

vector pre(m_c);

for (int i = 0; i < m_c; i++)

{

prei = vPreSumi + 1 - vPreSum0;

}

for(int i = 1 ; i < k ; i++ )

{

vector dp(m_c,-1);

int k = i ;

for (int j = i; j < m_c; j++)

{

k--;

int iMax = INT_MAX;

#define MaxCro (max(prek, vPreSumj + 1 - vPreSumk+1))

while ((k < j) && (MaxCro <= iMax))

{

iMax = MaxCro;

k++;

}

dpj = iMax;

}

pre.swap(dp);

}

return pre.back();

}

int m_c;

};

测试用例

template

void Assert(const vector& v1, const vector& v2)

{

if (v1.size() != v2.size())

{

assert(false);

return;

}

for (int i = 0; i < v1.size(); i++)

{

assert(v1i == v2i);

}

}

template

void Assert(const T& t1, const T& t2)

{

assert(t1 == t2);

}

int main()

{

vector nums = { 1,2,3,4,5,6 };

int k = 2;

auto res = Solution().splitArray(nums, k);

Assert(res, 11);

复制代码
 nums = { 1, 0, 3, 3, 0, 6 };
 k = 2;
 res = Solution().splitArray(nums, k);
Assert(res, 7);

nums = { 6,5,3,2,2,1 };
k = 5;
res = Solution().splitArray(nums, k);
Assert(res, 6);

nums = { 1,0,3,3,0,1 };
k = 5;
res = Solution().splitArray(nums, k);
Assert(res, 3);


//CConsole::Out(res);

}

2023年一月版:二分

class Solution {

public:

int splitArray(vector& nums, int k) {

int iMax = *std::max_element(nums.begin(), nums.end());

int iSum = std::accumulate(nums.begin(), nums.end(),0);

复制代码
	 int left = iMax-1, right = iSum;
	 while (left+1 < right)
	 {
		 int iMid = (left + right) / 2;
		 if (NeedK(nums, iMid) <= k)
		 {
			 right = iMid;
		 }
		 else
		 {
			 left = iMid;
		 }
	 }
	 return right;
 }

 int NeedK(const vector<int>& nums, int iMaxSum)
 {
	 int iNeedK = 1;
	 int iSum = 0;
	 for (const auto& n : nums)
	 {
		 if (iSum + n > iMaxSum)
		 {
			 iSum = n;
			 iNeedK++;
		 }
		 else
		 {
			 iSum+=n;
		 }
	 }
	 return iNeedK;
 }

};

2023年8月版也是二分

class Solution {

public:

int splitArray(vector& nums, int k) {

int iSum = std::accumulate(nums.begin(), nums.end(), 0);

int left = -1, r = iSum;

while (r > left + 1)

{

const auto mid = left + (r - left) / 2;

if (Is(nums, mid, k))

{

r = mid;

}

else

{

left = mid;

}

}

return r;

}

bool Is(const vector& nums, const int iMaxSum, int k)

{

k--;//可以分配的新组

int iHas = 0;

for (const auto& n : nums)

{

iHas += n;

if (iHas > iMaxSum)

{

k--;

iHas = n;

if (n > iMaxSum)

{

return false;

}

}

}

return k >= 0;

}

};

扩展阅读

视频课程

有效学习:明确的目标 及时的反馈 拉伸区(难度合适),可以先学简单的课程,请移步CSDN学院,听白银讲师(也就是鄙人)的讲解。
https://edu.csdn.net/course/detail/38771

如何你想快

速形成战斗了,为老板分忧,请学习C#入职培训、C++入职培训等课程
https://edu.csdn.net/lecturer/6176

相关下载

想高屋建瓴的学习算法,请下载《闻缺陷则喜算法册》doc版
https://download.csdn.net/download/he_zhidan/88348653

鄙人想对大家说的话
闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。
墨家名称的来源:有所得以墨记之。
如果程序是一条龙,那算法就是他的是睛

测试环境

操作系统:win7 开发环境: VS2019 C++17

或者 操作系统:win10 开发环境: VS2022 C++17

相关推荐
不会C语言的男孩27 分钟前
C++ Primer 第2章:变量和基本类型
开发语言·c++
云泽8082 小时前
C++ 可调用对象通关指南:深度解析 Lambda 表达式、function 包装器与 bind 绑定器
开发语言·c++·算法
wlsh152 小时前
Go 迭代器
算法
Tri_Function2 小时前
简单图论大学习
c++
语戚2 小时前
力扣 3161. 块放置查询:线段树解法(Java 实现)
java·算法·leetcode·面试·线段树·力扣·
lqqjuly3 小时前
C++ 完整知识体系—从基础语法到现代 C++23 的系统性总结
c++·c++23
CS创新实验室3 小时前
从顺序表到动态数组:数据结构的永恒基石与现代语言的优雅封装
数据结构·算法
王老师青少年编程3 小时前
信奥赛C++提高组csp-s之FHQ Treap
c++·csp·平衡树·信奥赛·csp-s·提高组·fhq treap
Black蜡笔小新4 小时前
自动化AI算法训练服务器DLTM训推一体化平台助力农业生产管理实现安全智能化
人工智能·算法·自动化
8Qi85 小时前
LeetCode 23. 合并 K 个升序链表 —— 小顶堆(PriorityQueue)
数据结构·算法·leetcode·链表·