前缀和-974.和可被k整除的子数组-力扣(LeetCode)

一、题目解析

1、子数组是数组中连续的部分

2、nums[-10^4,10^4],不能用滑动窗口优化

二、算法原理

解法1:暴力解法-枚举 O(N^2)

对于解法2的补充知识

1、同余定理

(a-b)/p=k......0->a%p=b%p
举例9%2=1,(2*4+1)%2=1%2,结合这个我们可以证明同余定理

证明:

(a-b)/p=k->a=b+p.k,对两边同时取余数,a%p=(b+p.k)%p=b%p

证毕

2、c++,java:[负数%正数]的结果以及修正

这里我们直接记结论,负数%正数=负数,-a%p=-a,对其进行修正,-a%p+p,最后在模上p,(-a%p+p)%p

解法2:前缀和+哈希表

我们需要找以最后位置为结尾的所有子数组,结合我们补充的知识,我们可以将问题转化

将问题转化为在[0,i-1]区间内找前缀和余数等于(sum%k+k)%k,对于哈希表unordered_map<int,int> hash,第一个int是前缀和余数,第二个int为出现的频率

细节问题:

1、前缀和插入数组的时机?

在判断完前缀和余数是否存在后,加入前缀和余数

2、由于前缀和余数会存进哈希表中,所以可以用一个变量记录新计算的前缀和余数,没必要开一个前缀和余数数组

3、如果整个前缀和所得余数为k,则hash[0]=1

4、什么时候统计子数组数目?

通过hash.count((sum%k+k)%k),如果哈希表中存在该余数,则该余数的频率可以加入到最终结果中

三、代码示例

解法2:

cpp 复制代码
class Solution {
public:
    int subarraysDivByK(vector<int>& nums, int k)
    {
        int sum = 0,ret = 0;
        unordered_map<int,int> hash;
        hash[0] = 1;
        for(auto& e : nums)
        {
            sum += e;
            if(hash.count((sum%k+k)%k))
            {
                ret += hash[(sum%k+k)%k];
            }

            hash[(sum%k+k)%k]++;
        }
        return ret;
    }
};

看到最后,如果对您有所帮助,还请点赞、收藏和关注,我们下期再见!

相关推荐
共享家952716 分钟前
LRU 缓存的设计与实现
开发语言·c++
夏鹏今天学习了吗21 分钟前
【LeetCode热题100(64/100)】搜索旋转排序数组
算法·leetcode·职场和发展
alphaTao38 分钟前
LeetCode 每日一题 2025/11/3-2025/11/9
windows·leetcode
草莓熊Lotso1 小时前
Linux 基础开发工具入门:软件包管理器的全方位实操指南
linux·运维·服务器·c++·人工智能·网络协议·rpc
小龙报1 小时前
算法通关指南:数据结构和算法篇 --- 队列相关算法题》--- 1. 【模板】队列,2. 机器翻译
c语言·开发语言·数据结构·c++·算法·学习方法·visual studio
晨非辰1 小时前
【数据结构初阶】--从排序算法原理分析到代码实现操作,参透插入排序的奥秘!
c语言·开发语言·数据结构·c++·算法·面试·排序算法
2301_795167205 小时前
玩转Rust高级应用 如何避免对空指针做“解引用”操作,在C/C++ 里面就是未定义行为
c语言·c++·rust
不染尘.10 小时前
2025_11_7_刷题
开发语言·c++·vscode·算法
似水এ᭄往昔10 小时前
【C++】--stack和queue
开发语言·c++
仰望—星空10 小时前
MiniEngine学习笔记 : CommandListManager
c++·windows·笔记·学习·cg·direct3d