前缀和(6)_和可被k整除的子数组_蓝桥杯

个人主页:C++忠实粉丝
欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 C++忠实粉丝 原创

前缀和(6)_和可被k整除的子数组

收录于专栏【经典算法练习
本专栏旨在分享学习算法的一点学习笔记,欢迎大家在评论区交流讨论💌

目录

[1. 题目链接 :](#1. 题目链接 :)

[2. 题目描述 :](#2. 题目描述 :)

[3. 解法(一维前缀和) :](#3. 解法(一维前缀和) :)

题目需要的前置知识:

[算法思路 :](#算法思路 :)

示例说明

[代码展示 :](#代码展示 :)

[结果分析 :](#结果分析 :)


1. 题目链接 :

OJ链接: 和可被k整除的子数组

2. 题目描述 :

给定一个整数数组 nums 和一个整数 k ,返回其中元素之和可被 k 整除的非空 子数组 的数目。

子数组 是数组中 连续 的部分。

示例 1:

复制代码
输入:nums = [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]

示例 2:

复制代码
输入: nums = [5], k = 9
输出: 0

提示:

  • 1 <= nums.length <= 3 * 104
  • -104 <= nums[i] <= 104
  • 2 <= k <= 104

3. 解法(一维前缀和) :

题目需要的前置知识:

同余定理:

如果(a - b) % n == 0, 那么我们就可以得到一个结论: a % n == b % n.用文字叙述就是,如果两个数相减的差能被n整除,那么这两个数取模的结果相同.

例如: (26 - 2) % 12 == 0,那么26 % 12 == 2 % 12.

C++中关于取模的结果,以及如何修正[负数取模]的结果

a. C++中关于负数的取模运算,结果是[把负数当成正数,取模之后的结果加上一个负号].

例如: -1 % 3 = -(1 % 3) = -1

b. 因为有负数,为了防止[出现负数]的结果,以(a % n + n) % n的形式输出保证为正.

例如: -1 % 3 = (-1 % 3 + 3) % 3 = 2;

算法思路 :

设i为数组中的任意位置,用sum[i]表示[0,i]区间内所有元素的和.

想知道有多少个[以i为结尾的可被k整除的子数组], 就要找到有多少个起始位置为x1,x2,x3......使得[x,i]区间内的所有元素的和可被k整除

设[0,x-1]区间内所有元素之和等于a,[0,i]区间内所有元素的和等于b,可得(b - a) % k == 0

由同余定理可得,[0,x - 1]区间与[0,i]区间内的前缀和同余.于是问题就变成:

找到在[0,i - 1]区间内,有多少前缀和的余数等于sum[i] % k的即可.

我们不需要真的初始化一个前缀和数组,因为我们只关心i位置之前,有多少个前缀和等于sum[i] - k.因此,我们仅需用一个哈希表,一边求当前位置的前缀和,一边存下之前每一种前缀和出现的次数.

示例说明

假设 nums = [4, 5, 0, -2, -3, 1]k = 5

  • 初始状态:dp = {0: 1}, sum = 0, ret = 0
  • 逐步计算:
    • 4sum = 4, r = 4, 更新 dp = {0: 1, 4: 1}
    • 5sum = 9, r = 4, ret += 1 (从 05),更新 dp = {0: 1, 4: 2}
    • 0sum = 9, r = 4, ret += 2 (从 0545),更新 dp = {0: 1, 4: 3}
    • -2sum = 7, r = 2, 更新 dp = {0: 1, 4: 3, 2: 1}
    • -3sum = 4, r = 4, ret += 3, 更新 dp = {0: 1, 4: 4, 2: 1}
    • 1sum = 5, r = 0, ret += 1, 更新 dp = {0: 2, 4: 4, 2: 1}

最后 ret 的值为 4,表示总共有 4 个子数组的和是 k 的倍数。

也就是说:我们在一段区间sum中,找到有多少个前缀和余数使得(sum % k + k) % k

因为(sum - 前缀和) % k 需要 == 0,根据同余定理,sum % k == 前缀和 % k

代码展示 :

cpp 复制代码
class Solution {
public:
    int subarraysDivByK(vector<int>& nums, int k) {
        unordered_map<int, int> dp(nums.size());
        dp[0] = 1; //0这个余数 
        int ret = 0, sum = 0;
        for(int i = 0; i < nums.size(); i++)
        {
            sum += nums[i];
            int r = (sum % k + k) % k;
            if(dp.count(r)) ret += dp[r];
            dp[r]++; 
        }
        return ret;
    }
};

初始化:

cpp 复制代码
unordered_map<int, int> dp(nums.size());
dp[0] = 1;
int ret = 0, sum = 0;

创建一个哈希表 dp 用于存储不同余数的出现次数。

初始化 dp[0] = 1,表示初始状态(和为0的情况)。

ret 用于存储结果(符合条件的子数组个数)。

sum 用于存储当前遍历的前缀和。

遍历数组:

cpp 复制代码
for (int i = 0; i < nums.size(); i++)
{
    sum += nums[i];
    int r = (sum % k + k) % k;
    if (dp.count(r)) ret += dp[r];
    dp[r]++;
}

使用 for 循环遍历数组 nums,计算前缀和。

对于当前的前缀和 sum,计算其对k的余数r。使用(sum % k + k) % k 确保余数为非负值。这是因为在 C++ 中,负数取模可能会返回负值。

如果 dp 中存在该余数r,则表示有一些前缀和的组合可以与当前前缀和形成可被k 整除的子数组,因此将这些组合的数量加到 ret 中。

最后,更新 dp 中余数r 的出现次数。

返回结果:

返回结果 ret,即符合条件的子数组数量。

dp[0] 的初始值

初始状态: 在代码的开头,我们将 dp[0] 初始化为 1,这意味着在开始时我们视为有一个和为 0 的"虚拟"子数组。这个初始化是为了帮助处理前缀和等于k 的倍数的情况。
**有效子数组的统计:**dp[0] 代表的是当前前缀和为 0 的子数组的出现次数。当我们在遍历过程中遇到新的前缀和,并且发现其余数为 0 时,就意味着从起始位置到当前的位置有一个有效的子数组。

结果分析 :

时间复杂度和空间复杂度
时间复杂度 :O(n),其中n 是数组的长度,因为我们只遍历数组一次。
空间复杂度 :O(k),因为哈希表 dp 最多存储k 个不同的余数。

相关推荐
qq_4335545427 分钟前
C++ 面向对象编程:递增重载
开发语言·c++·算法
带多刺的玫瑰1 小时前
Leecode刷题C语言之切蛋糕的最小总开销①
java·数据结构·算法
巫师不要去魔法部乱说1 小时前
PyCharm专项训练5 最短路径算法
python·算法·pycharm
qystca2 小时前
洛谷 P11242 碧树 C语言
数据结构·算法
冠位观测者2 小时前
【Leetcode 热题 100】124. 二叉树中的最大路径和
数据结构·算法·leetcode
悲伤小伞2 小时前
C++_数据结构_详解二叉搜索树
c语言·数据结构·c++·笔记·算法
m0_675988233 小时前
Leetcode3218. 切蛋糕的最小总开销 I
c++·算法·leetcode·职场和发展
佳心饼干-5 小时前
C语言-09内存管理
c语言·算法
dbln5 小时前
贪心算法(三)
算法·贪心算法
songroom6 小时前
Rust: offset祼指针操作
开发语言·算法·rust