力扣每日一题:使数组和能被p整除

给你一个正整数数组 nums,请你移除 最短 子数组(可以为 ),使得剩余元素的 能被 p 整除。 不允许 将整个数组都移除。

请你返回你需要移除的最短子数组的长度,如果无法满足题目要求,返回 -1

子数组 定义为原数组中连续的一组元素。

思路:

首先,这个题是要去掉某个子数组,因此要考虑用到前缀和的思想。

其次,这里是尝试去掉一部分后,剩下的部分余p等于0,也就是说,剩下的部分,和去掉的部分,余数是相同的。

那么,我们假设数组的和余p为x,要去除的区间是l,r,则有x=(sumr-suml)%p,而我们在遍历数组时,可以通过一次O(N)的时间复杂度得到x,后续遍历过程中,O(1)的时间复杂度得到sumr,那么,如果我们可以快速定位suml,就可以很快的找到符合条件的子数组。那么,如何快速找到这个suml,我们可以发现,这里要的是最短子数组,那么,suml一定是所有能符合x=(sumr-suml)%p中,最大的那个l。因此我们可以每次记录当前的前缀和余p,去更新对应位置。然后,每次只需要快速查找是否有符合条件的那个l,有就更新最小数组长度。

然后,我们来说一下这里的suml,suml就是((sumr-x)%p+p)%p,这里是确保一定出来的是一个正数。

要注意的是,这里如果用数组记录前缀和余p,会出现内存爆了的情况。

复制代码
class Solution {
public:
    int minSubarray(vector<int>& nums, int p) {
        int n = nums.size();
        vector<int> sum(n + 1, 0);
        for (int i = 0; i < n; i++)
            sum[i + 1] = (sum[i] + nums[i])%p;

        int x = sum[n] % p; // 表示两个区间模k余mm  (b-a)mod p=mm   bmodp -amodp==mm
        if (x == 0)
            return 0;

        for (int i = 0; i < n; i++) {
            if (nums[i] % p == x)
                return 1;
        }

        unordered_map<int,int> mxlen;// 记录当前模p余i的最大坐标
        int ans = n;
        for (int i = 0; i <= n; i++) {
           mxlen[sum[i]]=i;
           auto it=mxlen.find( ((sum[i]-x)%p+p)%p  );
           if(it!=mxlen.end())
           ans=min(ans,i-it->second);
        }
        if (ans==n)
            return -1;
        return ans;
    }
};
相关推荐
noipp2 小时前
推荐题目:洛谷 P10907 [蓝桥杯 2024 国 B] 蚂蚁开会
c语言·c++·算法·编程·洛谷
程序员二叉2 小时前
【JUC】线程池全套深度详解|参数|流程|拒绝策略|调优|异常处理
java·开发语言·jvm·算法·面试·juc
青山木3 小时前
Hot 100 --- 轮转数组
java·数据结构·算法
徐小夕3 小时前
Loop Engineering 深度解析与实战指南(全网最全)
前端·算法·github
北域码匠4 小时前
SHA-1算法:安全哈希原理与应用解析
算法·c#·哈希算法
手写码匠5 小时前
手写 GraphRAG:从零实现图增强检索增强生成系统
人工智能·深度学习·算法·aigc
BomanGe15 小时前
NSK重载高刚性滚珠丝杠技术详解
经验分享·算法·规格说明书
Matrix_116 小时前
手机里的计算摄影:广角形变校正算法
人工智能·算法·智能手机·计算摄影
WBluuue6 小时前
数据结构与算法:有序表(二):跳表
数据结构·c++·算法·skiplist
不好听6137 小时前
深入理解链表:线性数据结构的另一面
javascript·数据结构