LeetCode算法心得——和可被 K 整除的子数组(前缀和+HashMap)

这篇具有很好参考价值的文章主要介绍了LeetCode算法心得——和可被 K 整除的子数组(前缀和+HashMap)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

大家好,我是晴天学长,同余定理的应用,需要的小伙伴可以关注支持一下哦!后续会继续更新的。


1) .和可被 K 整除的子数组

LeetCode算法心得——和可被 K 整除的子数组(前缀和+HashMap),算法,算法,leetcode,职场和发展
题目描述

给定一个整数数组 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]

示例 2:

输入: nums = [5], k = 9
输出: 0


2) .算法思路

前缀和+HashMap

解析:

我们需要求出满足条件的区间,见下图
LeetCode算法心得——和可被 K 整除的子数组(前缀和+HashMap),算法,算法,leetcode,职场和发展
我们需要找到满足,和为 K 的区间。我们此时 presum 是已知的,k 也是已知的,我们只需要找到 presum - k区间的个数,就能得到 k 的区间个数。但是我们在当前题目中应该怎么做呢?见下图。

LeetCode算法心得——和可被 K 整除的子数组(前缀和+HashMap),算法,算法,leetcode,职场和发展
我们在之前的例子中说到,presum[j+1] - presum[i] 可以得到 nums[i] + nums[i+1]+… nums[j],也就是[i,j]区间的和。

那么我们想要判断区间 [i,j] 的和是否能整除 K,也就是上图中紫色那一段是否能整除 K,那么我们只需判断

(presum[j+1] - presum[i] ) % k 是否等于 0 即可,

我们假设 (presum[j+1] - presum[i] ) % k == 0;则

presum[j+1] % k - presum[i] % k == 0;

presum[j +1] % k = presum[i] % k ;

我们 presum[j +1] % k 的值 key 是已知的,则是当前的 presum 和 k 的关系,我们只需要知道之前的前缀区间里含有相同余数 (key)的个数。则能够知道当前能够整除 K 的区间个数。见下图
LeetCode算法心得——和可被 K 整除的子数组(前缀和+HashMap),算法,算法,leetcode,职场和发展
我们看到上面代码中有一段代码是这样的
int key = (presum % K + K) % K;
这是为什么呢?不能直接用 presum % k 吗?

这是因为当我们 presum 为负数时,需要对其纠正。纠正前(-1) %2 = (-1),纠正之后 ( (-1) % 2 + 2) % 2=1 保存在哈希表中的则为 1.则不会漏掉部分情况,例如输入为 [-1,2,9],K = 2如果不对其纠正则会漏掉区间 [2] 此时 2 % 2 = 0,符合条件,但是不会被计数。

那么这个题目我们可不可以用数组,代替 map 呢?当然也是可以的,因为此时我们的哈希表存的是余数,余数最大也只不过是 K-1所以我们可以用固定长度 K 的数组来模拟哈希表。


3) .算法步骤

1.创建一个HashMap对象 map 用于存储余数与对应出现次数的映射关系。
2.初始化变量 PerSum 为0,表示当前位置的前缀和。
3.初始化变量 answer 为0,表示满足条件的子数组个数。
4.将余数为0的初始情况放入 map 中,即前缀和为0的情况,出现次数为1。
5.使用一个循环遍历数组中的元素。
6.将当前元素的值累加到 PerSum 中,即计算当前位置的前缀和。
7.使用同余定理,计算当前前缀和除以k的余数,由于负数取模的结果可能为负数,为8.确保结果为非负整数,需要进行加k再取模的操作。
9.检查 map 中是否包含当前余数,如果是,则将对应的出现次数加到 answer 中。
10.将当前余数和对应的出现次数放入 map 中,更新映射关系。
11.循环结束后,返回 answer,表示满足条件的子数组个数。


4).代码示例

class Solution {
    public int subarraysDivByK(int[] nums, int k) {
         Map<Integer, Integer> map = new HashMap<Integer, Integer>();
            long PerSum = 0;
            int answer = 0;
            map.put(0, 1);
            for (int i = 0; i < nums.length; i++) {
                PerSum += nums[i];
                // 同余定理,因为不符合分配律
                int mod = (int) ((PerSum%k+k)%k);
                if (map.containsKey(mod)) {
                    answer+=map.get(mod);
                }
                map.put(mod, map.getOrDefault(mod, 0) + 1);
            }
            return answer;
    }
}

5).总结

  • 同余定理的运用。

试题链接:文章来源地址https://www.toymoban.com/news/detail-733183.html

到了这里,关于LeetCode算法心得——和可被 K 整除的子数组(前缀和+HashMap)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • LeetCode——可被三整除的偶数的平均值

    LeetCode——可被三整除的偶数的平均值

    #全国科技者工作日—为创新和未来而努力# 目录 1、题目  2、题目解读  3、代码 2455. 可被三整除的偶数的平均值 - 力扣(Leetcode) 给你一个由正整数组成的整数数组  nums  ,返回其中可被  3  整除的所有偶数的平均值。 注意: n  个元素的平均值等于  n  个元素  求和  

    2024年02月09日
    浏览(16)
  • Leetcode【1010】总持续时间可被 60 整除的歌曲

    最近忙于毕设,leetcode也就随缘刷刷每日一题了。 在歌曲列表中,第 i 首歌曲的持续时间为 time[i] 秒。 返回其总持续时间(以秒为单位)可被 60 整除的歌曲对的数量。形式上,我们希望下标数字 i 和 j 满足 i j 且有 (time[i] + time[j]) % 60 == 0 。 示例1: 来源:力扣(LeetCode) 链

    2024年02月03日
    浏览(25)
  • 【算法|滑动窗口No.1】leetcode209. 长度最小的子数组

    【算法|滑动窗口No.1】leetcode209. 长度最小的子数组

    个人主页:兜里有颗棉花糖 欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 兜里有颗棉花糖 原创 收录于专栏【手撕算法系列专栏】【LeetCode】 🍔本专栏旨在提高自己算法能力的同时,记录一下自己的学习过程,希望对大家有所帮助 🍓希望我们一起努力、成长,共同进步。

    2024年02月08日
    浏览(12)
  • LeetCode算法心得——k-avoiding 数组的最小总和(标记数组)

    LeetCode算法心得——k-avoiding 数组的最小总和(标记数组)

    大家好,我是晴天学长,这是一个细节题和一部分的思维题哈! 2) .算法思路 k-avoiding 数组的最小总和 1,填充一个1到n 的Boolean的数组 要n个数,但是数组大小不能确定。 所以建立1000的大小。 2.遍历筛选,如果数组中有这个的话,标记为false。 3.监测是否是false,true就sum++(前

    2024年02月11日
    浏览(10)
  • 和为 K 的子数组——前缀和+哈希

    和为 K 的子数组——前缀和+哈希

    题目链接:力扣 注意:此题不能使用滑动窗口,因为数组中可能会出现负数。也就是说右指针向后移1位不能保证区间会增大,左指针向后移1位也不能保证区间和会减小。给定左右指针的位置没有二段性 已知sum[i]是从nums[0~i]的和,sum[i-1]是nums[0~i-1]的和 则有 sum[i] - sum[i-1] =

    2024年02月16日
    浏览(8)
  • 560. 和为 K 的子数组【哈希、前缀和】

    560. 和为 K 的子数组【哈希、前缀和】

    560. 和为 K 的子数组 给你一个整数数组 nums 和一个整数 k ,请你统计并返回 该数组中和为 k 的连续子数组的个数 。 示例 1: 输入:nums = [1,1,1], k = 2 输出:2 示例 2: 输入:nums = [1,2,3], k = 3 输出:2 提示: 1 = nums.length = 2 * 104 -1000 = nums[i] = 1000 -107 = k = 107 C代码:哈希、前缀和

    2024年02月02日
    浏览(11)
  • java数据结构与算法刷题-----LeetCode209. 长度最小的子数组

    java数据结构与算法刷题-----LeetCode209. 长度最小的子数组

    java数据结构与算法刷题目录(剑指Offer、LeetCode、ACM)-----主目录-----持续更新(进不去说明我没写完): https://blog.csdn.net/grd_java/article/details/123063846 解题思路 代码:时间复杂度O(n).空间复杂度O(1)

    2024年01月21日
    浏览(47)
  • 【1262. 可被三整除的最大和】

    来源:力扣(LeetCode) 描述: 给你一个整数数组 nums ,请你找出并返回能被三整除的元素最大和。 示例 1: 示例 2: 示例 3: 提示: 1 = nums.length = 4 * 10 4 1 = nums[i] = 10 4 方法:贪心 + 正向思维 我们把数组中的数分成三部分 a, b, c,它们分别包含所有被 3 除余 0 , 1, 2 的数。显

    2024年02月10日
    浏览(9)
  • 【1015. 可被 K 整除的最小整数】

    【1015. 可被 K 整除的最小整数】

    来源:力扣(LeetCode) 描述: 给定正整数 k ,你需要找出可以被 k 整除的、仅包含数字 1 的最 小 正整数 n 的长度。 返回 n 的长度。如果不存在这样的 n ,就返回 -1 。 注意: n 不符合 64 位带符号整数。 示例 1: 示例 2: 示例 3: 提示: 1 = k = 10 5 方法:遍历 思路与算法 题

    2024年02月03日
    浏览(7)
  • [Kadane算法,前缀和思想]元素和最大的子矩阵

    题目描述 输入一个n级方阵,请找到此矩阵的一个子矩阵,此子矩阵的各个元素的和是所有子矩阵中最大的,输出这个子矩阵及这个最大的和。 关于输入 首先输入方阵的级数n, 然后输入方阵中各个元素。 关于输出 输出子矩阵, 最后一行输出这个子矩阵的元素的和。 例子输

    2024年02月03日
    浏览(10)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包