给定一个数组,值可以为正、负和0,请返回累加和小于等于k的最长子数组长度。
思路
申请两个辅助数组,分别为 min_v[N] 和 map_idx[N]
min_v[i] 用于记录从i开始,往右累加的最小和
map_idx[i]由于记录,得到最小和的右边界的下标
min_v[i]的记录规则就是,从优往左进行遍历,如果累加和一直<0 那么就一直累加下去,这个条件下,map_idx[i]就一直记录之前的右边界。
如果发现相加之后大于零,那么就另起炉灶,从自己开始作为右边界的开始,以及往右累加的最小和。
在计算最长长度时,通过计算每个min_v[i],可以进行整块的区间的相加,对应的下标也就可以通过map_idx[i]这个右边界进行跳跃。直到出现 sum >K 或者遍历结束。
以上面这种方式,我们从 第1,2,3,4 个位置分别开始计算,由于sum是累加和,
当从第2个开始时,直接去掉第一个的值就行,即 sum-=arr[1],然后再计算,看看之前sum >K的情况是否有所改变。
代码
#include<iostream>
#include<ctime>
#include<cstdlib>
#include<map>
using namespace std;
int N;
int arr[1000]={0};
void rand_arr(int N){
srand((int)time(0));
for(int i=0;i <N;i++){
arr[i] = rand()%11 -2; //[-2,5]
}
// for(int i=0;i <N;i++){
// arr[i] = 5;
// }
// arr[N-1] = 2;
}
void printarr(int N){
for(int i=0;i <N;i++){
printf("%d ",arr[i]);
}
printf("\n");
}
void init(int N){
rand_arr(N);
printarr(N);
}
//保存当前下标范围内的最小和
int min_v[1000]={0};
//记录当前下标和右边界
map<int,int> map_idx;
int main() {
N = 10;
init(N);
int K = 2;
map_idx[N-1] =N-1;
min_v[N-1] = arr[N-1];
//从后往前累加
for(int i=N-2;i>=0;i--){
if(min_v[i+1]<0){
min_v[i] = min_v[i+1] + arr[i];
map_idx[i] = map_idx[i+1];
}else{
min_v[i] = arr[i];
map_idx[i] = i;
}
}
//记录长度
int len =0;
//记录目前右边界
int end=0;
//记录目前的sum值
int sum=0;
//因为右边界只会往右边挪,并且sum可以一直累加
for(int i=0;i<N;i++){
while(sum+min_v[end]<=K && end<N){
sum+=min_v[end];
end = map_idx[end]+1;
}
//记录最大len
len = max(len,end-i);
//如果 end 和 i 相等的话,说明没有进入循环,即 sum根本没有加
sum -= end>i? arr[i]:0;
//保持 end >=i;
end = max(end,i+1);
}
printf("%d\n",len);
return 0;
}