和可被k整除的子数组 ---- 前缀和

题目链接

题目:

分析:

  • 补充知识
    1. 同余定理:  (a-b) % p = 0即a-b能被p整除, ==> a % p = b % p
    2. c++, java中 [负数 % 正数] 的结果是负数, 想要得到正确结果 ==> (a%p+p)%p
  • 这道题和<和为k的子数组>类似, 利用前缀和的思想, 计算以i结尾的所有子数组, 前缀和为sum[i] 
  • 想要找到和能被k整除的数组, 我们只需要找到在[0,i-1] 区间内, 找到有多少个前缀和能被k整除, 如图:
    sum[i] - x 这段区间和能被k整除, 即 (sum[i] - x)%k = 0, 根据同余定理, sum[i] % k = x % k, 所以问题就变成, 在[0,i-1] 中找有多少个前缀和的余数 = sum[i] % k, 但是前缀和有可能为负数, 所以c++和java要修改成(sum[i] % k + k) % k
  • 需要统计前缀和能被k整除的个数, 所以需要用哈希表, 存放(sum[i] % k + k) % k, 及个数
  • 细节:
    1. 前缀和加入哈希表的时机?
    因为要计算[0,i-1] 区间找有多少个前缀和的余数 = (sum[i] % k + k) % k, 所以要先更新有多少个前缀和的余数 = (sum[i] % k + k) % k 的个数, 再将sum[i]放到哈希表中
    2. 不用真的创建一个前缀和数组
    因为我们只需要记录和能被k整除的个数, 不需要知道下标等其他信息, 所以我们用一个变量sum记录前缀和即可, sum每次更新
    3. 如果整个前缀和等于k?
    如果从0到i的位置的和能被k整除, 那么在[0,i-1] 的位置寻找前缀和应该是0, 也是一种情况, 但是哈希表中存的是前缀和的余数, 所以哈希表中应该添加(0 % k,1)

代码:

class Solution {
    public int subarraysDivByK(int[] nums, int k) {
        Map<Integer, Integer> hash = new HashMap<>();
        hash.put(0 % k, 1);
        int sum = 0;
        int ret = 0;
        for (int i = 0; i < nums.length; i++) {
            sum += nums[i];//前缀和
            int r = (sum % k + k) % k;//前缀和的余数
            ret += hash.getOrDefault(r, 0);
            hash.put(r, hash.getOrDefault(r, 0) + 1);
        }
        return ret;
    }
}
  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值