博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
[LeetCode] Count of Range Sum 区间和计数
阅读量:6508 次
发布时间:2019-06-24

本文共 3136 字,大约阅读时间需要 10 分钟。

Given an integer array nums, return the number of range sums that lie in [lower, upper] inclusive.

Range sum S(i, j) is defined as the sum of the elements in nums between indices i and j (i ≤ j), inclusive.

Note:

A naive algorithm of O(n2) is trivial. You MUST do better than that.

Example:

Given nums = [-2, 5, -1]lower = -2upper = 2,
Return 3.
The three ranges are : [0, 0][2, 2][0, 2] and their respective sums are: -2, -1, 2.

Credits:

Special thanks to  for adding this problem and creating all test cases.

这道题给了我们一个数组,又给了我们一个下限和一个上限,让我们求有多少个不同的区间使得每个区间的和在给定的上下限之间。这道题的难度系数给的是Hard,的确是一道难度不小的题,题目中也说了Brute Force的方法太Naive了,那么我们只能另想方法了。To be honest,这题完全超出了我的能力范围,所以我也没挣扎了,直接上网搜大神们的解法啦。首先根据前面的那几道类似题,和的解法可知类似的区间和的问题一定是要计算累积和sum的,其中sum[i] = nums[0] + nums[1] + ... + nums[i],对于某个i来说,只有那些满足 lower <= sum[i] - sum[j] <= upper 的j能形成一个区间[j, i]满足题意,那么我们的目标就是来找到有多少个这样的j (0 =< j < i) 满足 sum[i] - upper =< sum[j] <= sum[i] - lower,我们可以用C++中由红黑树实现的multiset数据结构可以对其中数据排序,然后用upperbound和lowerbound来找临界值。lower_bound是找数组中第一个不小于给定值的数(包括等于情况),而upper_bound是找数组中第一个大于给定值的数,那么两者相减,就是j的个数,参见代码如下:

解法一:

class Solution {public:    int countRangeSum(vector
& nums, int lower, int upper) { int res = 0; long long sum = 0; multiset
sums; sums.insert(0); for (int i = 0; i < nums.size(); ++i) { sum += nums[i]; res += distance(sums.lower_bound(sum - upper), sums.upper_bound(sum - lower)); sums.insert(sum); } return res; } };

我们再来看一种方法,这种方法的思路和前一种一样,只是没有STL的multiset和lower_bound和upper_bound函数,而是使用了Merge Sort来解,在混合的过程中,我们已经给左半边[start, mid) 和右半边 [mid, end) 排序了。当我们遍历左半边,对于每个i,我们需要在右半边找出k和j,使其满足:

j是第一个满足 sums[j] - sums[i] > upper 的下标

k是第一个满足 sums[k] - sums[i] >= lower 的下标 

那么在[lower, upper]之间的区间的个数是j - k,同时我们也需要另一个下标t,用来拷贝所有满足sums[t] < sums[i]到一个寄存器Cache中以完成混合排序的过程。(注意这里sums可能会整型溢出,我们使用长整型long long代替)。

解法二:

class Solution {public:    int countRangeSum(vector
& nums, int lower, int upper) { vector
sums(nums.size() + 1, 0); for (int i = 0; i < nums.size(); ++i) { sums[i + 1] = sums[i] + nums[i]; } return countAndMergeSort(sums, 0, sums.size(), lower, upper); } int countAndMergeSort(vector
&sums, int start, int end, int lower, int upper) { if (end - start <= 1) return 0; int mid = start + (end - start) / 2; int cnt = countAndMergeSort(sums, start, mid, lower, upper) + countAndMergeSort(sums, mid, end, lower, upper); int j = mid, k = mid, t = mid; vector
cache(end - start, 0); for (int i = start, r = 0; i < mid; ++i, ++r) { while (k < end && sums[k] - sums[i] < lower) ++k; while (j < end && sums[j] - sums[i] <= upper) ++j; while (t < end && sums[t] < sums[i]) cache[r++] = sums[t++]; cache[r] = sums[i]; cnt += j - k; } copy(cache.begin(), cache.begin() + t - start, sums.begin() + start); return cnt; }};

 本文转自博客园Grandyang的博客,原文链接:,如需转载请自行联系原博主。

你可能感兴趣的文章
手把手教你将单机游戏改造成对战网游(附详细教程)
查看>>
React-Native flex 布局使用总结
查看>>
基于 Docker for MAC 的 Kubernetes 本地环境搭建与应用部署
查看>>
单例模式--SingleTon
查看>>
Choerodon猪齿鱼直播分享 | 华润置地中台转型实践分享
查看>>
iOS自定义UIPageControl
查看>>
来,搞个侧栏导航
查看>>
结合 Laravel 初步学习 GraphQL
查看>>
05.java多线程问题
查看>>
新功能:在负载均衡SLB控制台上查看DDoS安全防护阈值
查看>>
项目集成springboot【JWordpres前台项目实战】
查看>>
小知识三、USerDefault我换个姿势来实现
查看>>
JavaWeb项目架构之NFS文件服务器
查看>>
Activity事件分发机制
查看>>
设计模式快速学习(三)单例模式
查看>>
原生JS操作DOM
查看>>
利用CSS改变图片颜色的100种方法!
查看>>
《Redis开发与运维》笔记
查看>>
修改input type=file 默认样式
查看>>
函数防抖和函数节流
查看>>