LeetCode 523. Continuous Subarray Sum 解题报告

本文探讨了LeetCode上第523题“连续子数组和”的解决方案,介绍了如何检查给定数组中是否存在连续子数组其元素之和为k的倍数。文中详细解释了一种时间复杂度为O(n)的有效算法,该算法利用了数学特性,通过跟踪累加和的余数来快速判断。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

LeetCode 523. Continuous Subarray Sum 解题报告

题目描述

Given a list of non-negative numbers and a target integer k, write a function to check if the array has a continuous subarray of size at least 2 that sums up to the multiple of k, that is, sums up to n*k where n is also an integer.


示例

Example 1:
example1

Example 2:
example2


注意事项

1.The length of the array won’t exceed 10,000.
2.You may assume the sum of all the numbers is in the range of a signed 32-bit integer


解题思路

我的思路:

这道题是求给定的一个序列中是否存在一个连续子序列,满足子序列元素之和为k的倍数。最简单的办法,当然是暴力破解,通过二重循环,第一个重循环枚举每一个起点位置,第二重循环遍历以当前起点位置开始,各个长度的连续子序列,一旦找到和为K的倍数就直接返回true,如果一直都没找到就直接返回false,唯一需要注意的地方是几个特殊的测试用例,如nums = [1, 2], k = 0和nums = [0,0], k = 0。所以下面实现代码中的判断的条件才会略显复杂,时间复杂度是O(n2),空间复杂度是O(1)。
当然这道题可以用动态规划的思路去做,但是实现的时会发现时间复杂度接近O(n2),而空间复杂度比暴力破解更糟糕,会是O(n),所以就不贴出来我自己实现的动态规划的代码,如果有更好的动态规划的实现方式欢迎评论告知。

参考思路:

在讨论里有个大神给出了时间复杂度是O(n)的解法,他的思路非常巧妙,用了数学上的知识,下面给出他的解法的原理:
假设:

a[i]+a[i+1]+...+a[j]=n1k+q;

如果存在一个n
n>ja[i]+a[i+1]+...+a[j]+...+a[n]=n2k+q;

那么
a[j+1]+...+a[n]=(n2n1)k

因此利用这一结果,可以从序列第一个元素开始遍历,不断累加上当前的元素,并求出当前和除以k后的余数,用一个映射记录该余数出现时的下标,如果同一个余数出现了两次,并且两次出现的下标之差大于1,那么就表示在这两个坐标之间的元素之和是k的倍数,因此就可以返回true,否则最后返回false。
需要注意的两个地方:
1. k可能取0,所以只有当k不为0时才对当前的和求余,同时针对于nums = [0, 0], k = 0的情况,需要添加一个初始映射(0, -1)来确保结果的正确。
2. 下标之差至少为2才算是正确的情况,因为题目要求子序列长度至少为2,以上面的例子就是n至少等于j+2。
具体实现见下面参考代码。

代码

我的代码

class Solution {
public:
    bool checkSubarraySum(vector<int>& nums, int k) {
        if (nums.size () < 2)
            return false;

        int sum = 0;

        for (int i = 0; i < nums.size () - 1; i++) {
            sum += nums[i];
            for (int j = i + 1; j < nums.size (); j++) {
                sum += nums[j];
                if ((k != 0 && sum % k == 0) ||(k == 0 && sum == 0))
                    return true;
            }
            sum = 0;
        }

        return false;
    }
};

参考代码:

class Solution {
public:
    bool checkSubarraySum(vector<int>& nums, int k) {
        map<int, int> m;
        map<int, int>::iterator itr = m.end();
        int sum = 0;

        m[0] = -1;
        for (int i = 0; i < nums.size(); i++) {
            sum += nums[i];

            if (k)
                sum %= k;

            itr = m.find(sum);

            if (itr != m.end()) {
                if (i - itr->second > 1)
                    return true;
            }
            else
                m[sum] = i;
        }

        return false;
    }
};

总结

这道题是LeetCode Weekly Contest 21的第二道题,子序列和问题有很多变种,和为k倍数问题是其中一种,题目难度不大,可以用各种方式去做,只要注意k取0 的情况就行。
很多简单的数学性质都是一种很好的算法,所以以后做题时真的除了考虑数据结构外,还得多想想涉及到数学性质,说不定会有更好的解法。这周会再做几道分治相关的题目,坚持刷题,加油~

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值