首页 > 编程语言 >C++算法前缀和的应用:分割数组的最大值的原理、源码及测试用例

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

时间:2023-10-24 16:03:33浏览次数:30  
标签:vector const nums int res vPreSum C++ 测试用例 源码


分割数组的最大值

二分 过些天整理基础知识

题目

给定一个非负整数数组 nums 和一个整数 m ,你需要将这个数组分成 m 个非空的连续子数组。
设计一个算法使得这 m 个子数组各自和的最大值最小。
示例 1:
输入:nums = [7,2,5,10,8], m = 2
输出:18
解释:
一共有四种方法将 nums 分割为 2 个子数组。
其中最好的方式是将其分为 [7,2,5] 和 [10,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 <= nums[i] <= 10^6
1 <= m <= min(50, nums.length)

解法一:暴力解法

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

解法二:滑动窗口

假定j的j1是x,则当j增加时,x不变或增加。 当j++,vMaxSum[i-1][j’]不变,vPreSum[j+1] - vPreSum[j’] 增加。下面用因果表来证明。令L(j,x)= vMaxSum[i-1][x] R(j,x) = vPreSum[j+1] - vPreSum[x] |。
如果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++)
 {
 pre[i] = vPreSum[i + 1] - vPreSum[0];
 }
 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(pre[k], vPreSum[j + 1] - vPreSum[k+1]))
 while ((k < j) && (MaxCro <= iMax))
 {
 iMax = MaxCro;
 k++;
 }
 dp[j] = 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(v1[i] == v2[i]);
 }
 }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;
 }
 };

扩展阅读


相关下载

想高屋建瓴的学习算法,请下载《闻缺陷则喜算法册》doc版

鄙人想对大家说的话

闻缺陷则喜是一个美好的愿望,早发现问题,早修改问题,给老板节约钱。

墨家名称的来源:有所得以墨记之。

如果程序是一条龙,那算法就是他的是睛

测试环境

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

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


标签:vector,const,nums,int,res,vPreSum,C++,测试用例,源码
From: https://blog.51cto.com/u_15724537/8005220

相关文章

  • C++算法:给表达式添加运算符
    题目给定一个仅包含数字0-9的字符串num和一个目标值整数target,在num的数字之间添加二元运算符(不是一元)+、-或*,返回所有能够得到target的表达式。注意,返回表达式中的操作数不应该包含前导零。示例1:输入:num=“123”,target=6输出:[“1+2+3”,“123......
  • C++数位算法:数字1的个数
    题目给定一个整数n,计算所有小于等于n的非负整数中数字1出现的个数。示例1:输入:n=13输出:6示例2:输入:n=0输出:0提示:0<=n<=1092023年1月版classSolution{public:intcountDigitOne(intn){intiNum=0;intiMul=1;for(inti=0;i<9;i++){......
  • C++桶排序算法的应用:存在重复元素 III
    题目给你一个整数数组nums和两个整数indexDiff和valueDiff。找出满足下述条件的下标对(i,j):i!=j,abs(i-j)<=indexDiffabs(nums[i]-nums[j])<=valueDiff如果存在,返回true;否则,返回false。示例1:输入:nums=[1,2,3,1],indexDiff=3,valueDiff=0输出......
  • C++前缀和算法应用:矩形区域不超过 K 的最大数值和
    题目给你一个mxn的矩阵matrix和一个整数k,找出并返回矩阵内部矩形区域的不超过k的最大数值和。题目数据保证总会存在一个数值和不超过k的矩形区域。示例1:输入:matrix=[[1,0,1],[0,-2,3]],k=2输出:2解释:蓝色边框圈出来的矩形区域[[0,1],[-2,3]]的数值和是......
  • 时间复杂度O(40n*n)的C++算法:修改图中的边权
    1.12.1.题目给你一个n个节点的无向带权连通图,节点编号为0到n-1,再给你一个整数数组edges,其中edges[i]=[ai,bi,wi]表示节点ai和bi之间有一条边权为wi的边。部分边的边权为-1(wi=-1),其他边的边权都为正数(wi>0)。你需要将所有边权为-1的边都修改为范......
  • 现代化的LIS实验室信息管理系统应该这样做!全套源码分享
    开发一款完整的实验室药剂/设备管理系统,包括安卓、IOS用户端和网页管理端,用于管理实验室的物资借用跟踪。可以给每一个物资生成专属二维码,一物一码,扫码借用或者归还。支持内网部署保护数字资产安全。高级功能:支持借用超时消息提醒支持超时自动归还支持导出报表统计、每日实......
  • 在C++中,互斥变量(std::mutex)是用于保护共享资源的重要工具,但它们确实有一些局限性,其中
    在C++中,互斥变量(std::mutex)是用于保护共享资源的重要工具,但它们确实有一些局限性,其中之一是无法保证包含指针的区域的多线程安全。这是因为互斥锁本质上只能保护它们所保护的代码块,而不会考虑指针指向的数据。下面是一些与互斥锁和指针相关的常见问题和注意事项:共享数据的复制:......
  • 成品直播源码推荐,用JNI生成so文件,加密解密需要的hascode生成代码
    成品直播源码推荐,用JNI生成so文件,加密解密需要的hascode生成代码try{      PackageInfopackageInfo=getPackageManager().getPackageInfo(getPackageName(),PackageManager.GET_SIGNATURES);      Signature[]signs=packageInfo.signatures; ......
  • C++_Cmake的使用
    C++系统版本、软件依赖版本、组件LSB(全称:LinuxStandardsBase)LSBsharedobjectELF是ExecutableLinkableFormat的缩写,是Linux的链接、可执行、共享库的格式标准,COFF:CommonObjectCOFF(通用对象文件格式) 编译器:简单构建gcc编译流程分为4个步骤,分......
  • C++初识(续篇)
    1.2注释作用:在代码中加一些说明和解释,方便自己或其他程序员阅读代码两中格式单行注释:通常放在一行代码的上方,或者一条语句的末尾,对该行代码说明//这样的是单行注释多行注释:通常放在一段代码的上方,对该段代码做整体说明/*这种的是多行注释可以写好多行*/......