LeetCode 974. 和可被 K 整除的子数组 | Python
974. 和可被 K 整除的子数组
题目来源:力扣(LeetCode)https://leetcode-cn.com/problems/subarray-sums-divisible-by-k
题目
给定一个整数数组 A,返回其中元素之和可被 K 整除的(连续、非空)子数组的数目。
示例:
输入:A = [4,5,0,-2,-3,1], K = 5
输出:7
解释:
有 7 个子数组满足其元素之和可被 K = 5 整除:
[4, 5, 0, -2, -3, 1], [5], [5, 0], [5, 0, -2, -3], [0], [0, -2, -3], [-2, -3]
提示:
- 1 <= A.length <= 30000
- -10000 <= A[i] <= 10000
- 2 <= K <= 10000
解题思路
思路:前缀和
首先这里要提前说明一下,关于除法取整,Python 采用的是向下取整。这里可能跟预想出现偏离的情况,比如,除数为负数的情况。看以下示例:
>>> 10 // 3
3
>>> 10 % 3
1
>>> 10 // -3
-4
>>> 10 % -3
-2
这里可以看到,关于正数取整,取模跟预想都不会有太大的偏离。但是对于负数,出现的结果可能就跟预想的不太一样。这里是因为前面说,Python 采用的是向下取整。
对于 10 ÷ -3 = -3.3333,这时向下取整就会得到 -4,那么余数就是 -2。这就是为什么会出现上面结果的原因。这部分内容仅做一些提示。
在本题当中 K,这里是除数,题目中说明 【2 <= K <= 10000】,所以不会有上面所述的情况。但是 Python 当中负数取余所得的结果是正数,这里跟一些其他编程语言不同。有些语言负数取余的结果是负数,所以要额外进行处理。
本篇幅使用的是前缀和的思想。关于前缀和,大致说明下。
如果要求前 i 项的和,那么:
P[i] = A[0]+A[1]+...+A[i]
相应的前 i-1 项的和为:
P[i-1] = A[0]+A[1]+...+A[i-1]
那么,A[i] 的值也可以表示为
A[i] = P[i] - P[i-1]
那么相应的,如果要计算 i 项到 j 项连续子数组的和,也可用写成如下的形式:
sum[i...j] = P[j] - P[i-1],其中 (0 < i < j)
那么题目中要求,判断子数组的和是否能够被 K 整除,现在就等同于判断 (P[j]-P[i-1]) mod K == 0 是否成立。
在这里,额外提及一个定理:同余定理。
同余定理:给定一个正整数 m,如果两个整数 a 和 b 满足 a-b 能够被 m 整除,即 (a-b)/m 得到一个整数,那么就称整数 a 与 b 对模 m 同余。
那么上面需要判断的式子也就可以转换为求 P[j] mod K == P[i-1] mod K 式子是否成立。
具体的方法:
- 维护哈希表,其中哈希表键为前缀和模 K 的值,值为出现的次数。
- 遍历数组每项,求当前前缀和模 K,存入哈希表中
- 当不存在表中,则将键跟值存入
- 存在时,对应键的值 +1
- 遍历的同时,进行统计,如果哈希表中存在 key 与当前前缀和模 k 的值相等时:
- 说明前面存在前缀和模 K 的值与此次计算的值相同
- 将满足条件的 key 出现的次数,累加到结果中
具体的代码实现如下。
代码实现
class Solution:
def subarraysDivByK(self, A: List[int], K: int) -> int:
# 这里是考虑前缀和本身被 K 整除的情况
hashmap = {0: 1}
pre_sum = 0
cnt = 0
for i in range(len(A)):
pre_sum += A[i]
# 取模
mod = pre_sum % K
# 这里使用字典的 get 方法
# 当存在相同的键时,累加到 cnt
if mod in hashmap:
cnt += hashmap[mod]
hashmap[mod] += 1
# 如果键在哈希表中,则次数加 1,
# 否则初始化为 1
else:
hashmap[mod] = 1
return cnt
实现结果
以上就是使用前缀和的思路,解决《974. 和可被 K 整除的子数组》问题的主要内容。
欢迎关注微信公众号《书所集录》