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

server/2024/10/9 3:22:14/

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

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

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

目录

1. 题目链接 :

2. 题目描述 :

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 = 0ret = 0
  • 逐步计算:
    • 加 4sum = 4r = 4, 更新 dp = {0: 1, 4: 1}
    • 加 5sum = 9r = 4ret += 1 (从 0 到 5),更新 dp = {0: 1, 4: 2}
    • 加 0sum = 9r = 4ret += 2 (从 0 到 5 和 4 到 5),更新 dp = {0: 1, 4: 3}
    • 加 -2sum = 7r = 2, 更新 dp = {0: 1, 4: 3, 2: 1}
    • 加 -3sum = 4r = 4ret += 3, 更新 dp = {0: 1, 4: 4, 2: 1}
    • 加 1sum = 5r = 0ret += 1, 更新 dp = {0: 2, 4: 4, 2: 1}

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

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

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

    代码展示 :

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;}
};

初始化:

unordered_map<int, int> dp(nums.size());
dp[0] = 1;
int ret = 0, sum = 0;

创建一个哈希表 dp 用于存储不同余数的出现次数。
初始化 dp[0] = 1,表示初始状态(和为0的情况)。
ret 用于存储结果(符合条件的子数组个数)。
sum 用于存储当前遍历的前缀和。

遍历数组:

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 个不同的余数。


http://www.ppmy.cn/server/127766.html

相关文章

WPF入门教学二十二 多线程与异步编程

在WPF&#xff08;Windows Presentation Foundation&#xff09;中&#xff0c;多线程和异步编程是非常重要的概念&#xff0c;因为它们可以帮助你创建响应性更好的应用程序。WPF的UI线程负责处理所有的用户界面操作&#xff0c;如果你的代码在UI线程上执行耗时操作&#xff0c…

使用Materialize制作unity的贴图,Materialize的简单教程,Materialize学习日志

Materialize 官网下载地址&#xff1a;http://boundingboxsoftware.com/materialize/ github源码地址&#xff1a;https://github.com/BoundingBoxSoftware/Materialize 下载地址&#xff1a;http://boundingboxsoftware.com/materialize/getkey.php 下载后解压运行exe即可 …

软件验证与确认实验一:静态分析

目录 1. 实验目的及要求.................................................................................................... 3 2. 实验软硬件环境.................................................................................................... 3 …

Python环境安装教程

文章目录 一、搭建Python环境1.官网下载Python2.安装Python3.检验是否安装成功 二、安装pip1.检验是否有pip2.pip升级3.模块安装4.检验模块是否安装成功5.番外&#xff1a;pip做了什么&#xff1f; 本教程是安装Windows环境下Python3.7.4 一、搭建Python环境 1.官网下载Python…

<Rust>iced库(0.13.1)学习之部件(二十九):button部件新增方法on_press_with,可传入闭包函数

前言 本专栏是学习Rust的GUI库iced的合集,将介绍iced涉及的各个小部件分别介绍,最后会汇总为一个总的程序。 iced是RustGUI中比较强大的一个,目前处于发展中(即版本可能会改变),本专栏基于版本0.12.1. 注:新版本已更新为0.13 概述 这是本专栏的第二十九篇,在新版本中…

【GeekBand】C++设计模式笔记5_Observer_观察者模式

1. “组件协作”模式 现代软件专业分工之后的第一个结果是“框架与应用程序的划分”&#xff0c;“组件协作”模式通过晚期绑定&#xff0c;来实现框架与应用程序之间的松耦合&#xff0c;是二者之间协作时常用的模式。典型模式 Template MethodStrategyObserver / Event 2.…

GO网络编程(四):海量用户通信系统2:登录功能核心【重难点】

目录 一、C/S详细通信流程图二、消息类型定义与json标签1. 消息类型定义2. JSON标签3.结构体示例及其 JSON 表示&#xff1a;4.完整代码与使用说明 三、客户端发送消息1. 连接到服务器2. 准备发送消息3. 创建 LoginMes 并序列化4. 将序列化后的数据嵌入消息结构5. 序列化整个 M…

C语言语句、语句分类及注释

文章目录 一、语句和语句分类二、注释&#x1f355;注释是什么&#xff1f;为什么写注释&#xff1f;1. /**/的形式2. //的形式3. 注释会被替换 三、随机数的生成1.rand函数2.srand函数3.time函数4.设置随机数的范围 四、C99中的变长数组五、问题表达式解析表达式1表达式2表达式…