资讯专栏INFORMATION COLUMN

327. Count of Range Sum

mj / 516人阅读

摘要:题目链接这题实际就是给定范围内的,的方法。注意一开始把加进去,考虑结果是的情况,还有要用型,以免会还是可以来做,要统计范围内的个数,就是用。

327. Count of Range Sum

题目链接:https://leetcode.com/problems...

这题实际就是给定范围内的range sum,divide and conquer的方法。一路计算prefixSum[0:i],并把结果放进tree里面,然后计算到prefixSum[0:j+1]的时候,找tree里面有没有满足条件的prefixSum[0:i],这里的条件是lower <= sum[0:j+1] - sum[0:i] <= upper,那么可知sum[0:j+1] - upper <= sum[0:i] <= sum[0:j+1] - lower,那么这个就一个recursion就好了。注意一开始把0加进去,考虑结果是sum[0:j]的情况,还有要用long型,以免sum会overflow

</>复制代码

  1. public class Solution {
  2. public int countRangeSum(int[] nums, int lower, int upper) {
  3. int n = nums.length;
  4. if(n == 0) return 0;
  5. // binary search tree
  6. Node root = new Node(0);
  7. int res = 0;
  8. long prefixSum = 0;
  9. for(int i = 0; i < n; i++) {
  10. prefixSum += nums[i];
  11. res += findNumInBound(root, lower, upper, prefixSum);
  12. insert(root, prefixSum);
  13. }
  14. return res;
  15. }
  16. private int findNumInBound(Node node, long low, long up, long sum) {
  17. // base case
  18. if(node == null) return 0;
  19. // range: sum - up <= node.val <= sum - low
  20. if(node.val < sum - up) return findNumInBound(node.right, low, up, sum);
  21. else if(node.val > sum - low) return findNumInBound(node.left, low, up, sum);
  22. else return 1 + findNumInBound(node.left, low, up, sum) + findNumInBound(node.right, low, up, sum);
  23. }
  24. private void insert(Node node, long value) {
  25. while(node != null) {
  26. if(node.val > value) {
  27. if(node.left == null) {
  28. node.left = new Node(value);
  29. break;
  30. }
  31. node = node.left;
  32. }
  33. else {
  34. if(node.right == null) {
  35. node.right = new Node(value);
  36. break;
  37. }
  38. node = node.right;
  39. }
  40. }
  41. }
  42. class Node {
  43. long val;
  44. Node left;
  45. Node right;
  46. Node(long val) { this.val = val; }
  47. }
  48. }

还是可以binary index tree来做,要统计sum[0:j+1] - upper <= sum[0:i] <= sum[0:j+1] - lower范围内的个数,就是用sum。参考博客:
http://bookshadow.com/weblog/...

</>复制代码

  1. public class Solution {
  2. public int countRangeSum(int[] nums, int lower, int upper) {
  3. int n = nums.length;
  4. if(n == 0) return 0;
  5. // prefix array
  6. long[] prefixSum = new long[n];
  7. for(int i = 0; i < n; i++) {
  8. prefixSum[i] = (i > 0 ? prefixSum[i-1] : 0) + nums[i];
  9. }
  10. long[] sorted = Arrays.copyOf(prefixSum, prefixSum.length);
  11. Arrays.sort(sorted);
  12. // binary index tree
  13. map = new HashMap();
  14. int idx = 1;
  15. for(long sum : sorted) {
  16. if(!map.containsKey(sum)) map.put(sum, idx++);
  17. }
  18. // build tree
  19. BIT t = new BIT(idx);
  20. int res = 0;
  21. for(int i = 0; i < n; i++) {
  22. int l = binarySearch(sorted, prefixSum[i] - upper - 1);
  23. int r = binarySearch(sorted, prefixSum[i] - lower);
  24. res += t.sum(r) - t.sum(l);
  25. if(prefixSum[i] >= lower && prefixSum[i] <= upper) res += 1;
  26. t.add(map.get(prefixSum[i]), 1);
  27. }
  28. return res;
  29. }
  30. Map map;
  31. // find the last element <= val
  32. private int binarySearch(long[] arr, long val) {
  33. int l = 0, r = arr.length - 1;
  34. while(l < r) {
  35. int mid = l + (r - l) / 2 + 1;
  36. if(arr[mid] <= val) l = mid;
  37. else r = mid - 1;
  38. }
  39. if(arr[l] > val) return 0;
  40. return map.get(arr[l]);
  41. }
  42. class BIT {
  43. int n;
  44. int[] tree;
  45. BIT(int n) { this.n = n; tree = new int[n]; }
  46. protected int sum(int i) {
  47. int res = 0;
  48. while(i > 0) {
  49. res += tree[i];
  50. i -= i & -i;
  51. }
  52. return res;
  53. }
  54. protected void add(int i, int val) {
  55. while(i < n) {
  56. tree[i] += val;
  57. i += i & -i;
  58. }
  59. }
  60. }
  61. }

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/66674.html

相关文章

  • leetcode327. Count of Range Sum

    摘要:接着计算所有子数组中元素的和,并判断是否位于数值区间内。因此,在对左右子数组进行排序后,以左子数组中的每一位作为开头,在右子数组中找到满足和区间的第一个值,和超过区间的第一个值。则二者的差即为横穿左右的满足条件的子数组个数。 题目要求 Given an integer array nums, return the number of range sums that lie in [lo...

    miya 评论0 收藏0
  • 几个有趣的算法题目

    摘要:统计元素个数排序第一步用来枚举和的大小,由题目可知,数组的长度。时间复杂度为数组长度,排序的时间为,枚举时间为,枚举时间为跨度,枚举全部元素时间为,因此算法的时间上界为实际情况下,由于剪枝等操作的存在,应优于这个时间。 本文首发 http://svtter.cn 最接近的数字 题目 一个K位的数N $$ (Kleq2000,Nleq10^{20}) $$ 找出一个比N大且最接近的数,这...

    honhon 评论0 收藏0
  • SICP Python 描述 2.3 序列

    摘要:序列不是特定的抽象数据类型,而是不同类型共有的一组行为。不像抽象数据类型,我们并没有阐述如何构造序列。这两个选择器和一个构造器,以及一个常量共同实现了抽象数据类型的递归列表。 2.3 序列 来源:2.3 Sequences 译者:飞龙 协议:CC BY-NC-SA 4.0 序列是数据值的顺序容器。不像偶对只有两个元素,序列可以拥有任意(但是有限)个有序元素。 序列在计算机科学中...

    AlexTuan 评论0 收藏0
  • 315. Count of Smaller Numbers After Self

    摘要:题目链接的题,用来做,这种求有多少的题一般都是。里多加一个信息表示以为的节点数。也可以做,因为是统计有多少的,其实就是求从最小值到的。的是,要做一个映射,把的值映射到之间。所以先把给一下,用一个来做映射。还有的方法,参考 315. Count of Smaller Numbers After Self 题目链接:https://leetcode.com/problems... divi...

    cnio 评论0 收藏0
  • [LeetCode] 4Sum & 4Sum II

    摘要:和方法一样,多一个数,故多一层循环。完全一致,不再赘述, 4Sum Problem Given an array S of n integers, are there elements a, b, c, and d in S such that a + b + c + d = target? Find all unique quadruplets in the array which ...

    sydMobile 评论0 收藏0

发表评论

0条评论

mj

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<