LeetCode: 327. Count of Range Sum

LeetCode: 327. Count of Range Sum

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:

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

解题思路 ——前缀和 + 分治

  1. 计算前缀和
  2. 根据前缀和递归分治的求解局部的 count
  3. 在分治的同时,对前缀和进行排序。这样方便在 O(N) 的复杂度计算同时有分治的两块的数据时,满足要求的序列

AC 代码

func judge(sub int, lower int, upper int) int {
    if sub <= upper && sub >= lower {
        return 1
    } else {
        return 0
    }
}

// [beg, end) 的 Count
func countRangeSumRef(frontSum []int, lower int, upper int) int {
    if len(frontSum) == 0 {
        return 0
    }
    if len(frontSum) == 1 {
        return judge(frontSum[0], lower, upper) 
    }
    
    // divide and conquer
    count := 0
    mid := len(frontSum) / 2
    count += countRangeSumRef(frontSum[:mid], lower, upper)
    count += countRangeSumRef(frontSum[mid:], lower, upper)
   
    // count
    j, k := mid, mid
    for i := 0; i < mid; i++ {
        for j < len(frontSum) && frontSum[j] - frontSum[i] < lower { j++ }
        for k < len(frontSum) && frontSum[k] - frontSum[i] <= upper { k++ }
        count += (k-j)
    }
    
    // sort
    iter := 0
    arr := make([]int, len(frontSum))
    copy(arr, frontSum)
    left, right := 0, mid
    for left < mid || right < len(frontSum) {
        if left >= mid {
            frontSum[iter] = arr[right]
            right++
        } else if right >= len(frontSum) {
            frontSum[iter] = arr[left]
            left++
        } else if arr[left] <= arr[right] {
            frontSum[iter] = arr[left]
            left++
        } else {
            frontSum[iter] = arr[right]
            right++
        }
        iter++
    }

    return count
}

func countRangeSum(nums []int, lower int, upper int) int {
    // 前缀和
    var frontSum []int
    for i := 0; i < len(nums); i++ {
        if i == 0 {
            frontSum =  append(frontSum, nums[i])
        } else {
            frontSum = append(frontSum, frontSum[i-1] + nums[i])
        }
    }

    return countRangeSumRef(frontSum, lower, upper)
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值