LeetCode subarray-sum-equals-k题解 前缀和+Hash表+枚举——线性做法

题意

给定一个数组,求连续的子数组的和为k的子数组个数。

思路

连续子数组的和sum[i,j]

s u m [ i , j ] = k = i j A k ( i &lt; j ) sum[i,j]=\sum_{k=i}^jA_k(i&lt;j) sum[i,j]=k=ijAk(i<j)
即数组第i个数到第j个数的子数组的和。
s u m [ i , j ] = s u m [ j ] s u m [ i 1 ] sum[i,j]=sum[j]-sum[i-1] sum[i,j]=sum[j]sum[i1]其中 s u m [ i ] sum[i] sum[i]是前i个数的前缀和。
另子数组的和固定为k,则右边两个数只需要每一个,就可以确定另一个数的取值。
假设我们把前一个sum的下标记为r,后一个记为l.
若我们枚举l,我们有了如下需求:
sum数组中,下标大于l且值为sum[l]+k的个数是多少?
因此我们可以控制l的枚举顺序为从后往前,并且用hash表维护自l往后的sum[i]中的所有<sum值,个数>以应对此查询并且快速更新hash表。


同理,可以从前往后枚举r,用hash表维护下标小于r的所有的<sum值,个数>……

源码

class Solution {
    public int subarraySum(int[] nums, int k) {
        int len = nums.length;
        //key:sum,value:the count of sum[i] where i<r
        Map<Integer,Integer> map = new HashMap<Integer, Integer>();
        map.put(0,1); // sum[-1]=0
        int count = 0;
        for (int r = 0,r_sum = 0;r < len; ++r) {
            r_sum += nums[r];
            int l_sum = r_sum-k;
            if (map.containsKey(l_sum))
                count += map.get(l_sum);
            if (map.containsKey(r_sum))
                map.put(r_sum,map.get(r_sum)+1);
            else
                map.put(r_sum,1);
        }
        return count;
    }
}

结果记录

  • 19ms
  • 82.65%
  • 40.1MB
  • 100.0%
全部评论

相关推荐

头像
昨天 14:28
长沙理工大学
刷算法真的是提升代码能力最快的方法吗?&nbsp;刷算法真的是提升代码能力最快的方法吗?
牛牛不会牛泪:看你想提升什么,代码能力太宽泛了,是想提升算法能力还是工程能力? 工程能力做项目找实习,算法也分数据结构算法题和深度学习之类算法
点赞 评论 收藏
分享
头像 会员标识
昨天 17:08
已编辑
牛客_产品运营部_私域运营
腾讯 普通offer 24k~26k * 15,年包在36w~39w左右。
点赞 评论 收藏
分享
我即大橘:耐泡王
点赞 评论 收藏
分享
牛客868257804号:九个中铁八个中建
点赞 评论 收藏
分享
点赞 收藏 评论
分享
牛客网
牛客企业服务