leedcode 刷题 - 除自身以外数组的乘积 - 和为 K 的子数组

news2024/12/25 23:47:23

I238. 除自身以外数组的乘积 - 力扣(LeetCode)

给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。

题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在  32 位 整数范围内。

请 不要使用除法,且在 O(n) 时间复杂度内完成此题。

示例 1:

输入: nums = [1,2,3,4]
输出: [24,12,8,6]
示例 2:

输入: nums = [-1,1,0,-3,3]
输出: [0,0,9,0,0]
 

提示:

2 <= nums.length <= 105
-30 <= nums[i] <= 30
保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在  32 位 整数范围内
 

进阶:你可以在 O(1) 的额外空间复杂度内完成这个题目吗?( 出于对空间复杂度分析的目的,输出数组 不被视为 额外空间。)

题目已经给的很明显了,就是使用前缀积 和 后缀积 的方式来求解,这种方式其实 是一个简化的 动态规划。

用两个数组,分别存储 从起始位置到 i 位置的乘积 和 从 末尾位置 到 i 位置的乘积;

上述两个结果对应的就是 f[i]  和  g[i] 。

递推公式:

f[i] = f[i - 1] * nums[i - 1];

g[i] = g[i + 1] * nums[i + 1];

f[0] 和 g[nums.size()] 都是需要自己手动算的,上述递推公式是算不出来的。

如果我们像计算题目描述的某个位置(比如是 i 位置)的 前缀积 和 后缀积的话,只需要计算        f[i] * g[i] 既可,因为 f[i] 表示的就是 i 之前的 所有积,g[i] 表示的就是 i 之后的所有的积。

完整代码:

class Solution {
public:
    vector<int> productExceptSelf(vector<int>& nums) {
        vector<int> f(nums.size(), 1);
        vector<int> g(nums.size(), 1);
        vector<int> ret(nums.size());

        for(int i = 1;i < nums.size(); i++) f[i] = f[i - 1] * nums[i - 1];
        for(int i = nums.size() - 2; i >= 0; i--) g[i] = g[i + 1] * nums[i + 1];
        for(int i = 0;i < nums.size();i++) ret[i] = g[i] * f[i];

        return ret;
}
};

I560. 和为 K 的子数组 - 力扣(LeetCode)

给你一个整数数组 nums 和一个整数 k ,请你统计并返回 该数组中和为 k 的子数组的个数 

子数组是数组中元素的连续非空序列。

示例 1:

输入:nums = [1,1,1], k = 2
输出:2
示例 2:

输入:nums = [1,2,3], k = 3
输出:2

 还是使用前缀和,我们可以使用 找到一个前缀和 sum[i] 就表示,从数组 0 号位置开始,到 i 位置的所有元素之和。

那么,我们只需要在这个区间当中找到 在 [0,i] 这个区间当中,某一个位置作为起始位置(假设为 j   ),到 i 位置,这个子数组的 元素之和等于k,那么 ,[0,j] 这个区间当中各个 元素之和就是           sum[i] - k;

 往后,只需要  i 往后寻找,就不会找漏。

但是,上述还是有一个问题,如果我们直接遍历 sum 数组的,来找到 j 这个位置的话,在加上 创建 sum 数组的时间复杂度,实际上,这个算法的整个 时间复杂度其实还不如 暴力解法。

所以,其实我们不用  循环一个一个 去找  j  位置,我们可以利用一个 hash 表来 代替 sum 存储的这些 前缀和的值,也就是代替 sum 存储 其中的每一个元素。

哈希表:  hash<前缀和值, 前缀和出现次数>

这样,如果我们 想 在  [0,i] 这个区间当中, 找到 j 这个位置,只需要 利用 hash 表的快速查找来查找 在当前hash 表当中有没有 sum[i] - k 这个前缀和,同时,利用 hash 表当中的 count()函数,可以快速查找,这个 sum[i] - k  出现次数。

关于前缀和加入hash 表的时机:
因为我们的前缀和算法,是要找的是 在 [0,i] 这个区间当中 ,有多少个 前缀和 等于sum[i] - k;   。

如果直接 在一开始就把 sum 计算出来,然后把 区间当中 前缀和 和 前缀和出现的次数加入到 hash 表当中是会计算到 i 后面的值。所以不行。

所以,我们在计算 i 位置之前,哈希表里面值存储 [0, i - 1] 位置的前缀和。

还有一种情况,当 当前的整个的 前缀和 等于 k 的话,那么,在上述的算法当中,其实我们是找不到这个情况的,因为 我们找到的是 等于 k 的子区间,这个子区间的起始位置上述说过了,是 j ,那么 满足   sum[i] - k; 的 区间就是 [0, j - 1], 那么在这个情况当中就是 [0, -1] 这个区间,这个区间是不存在的。

所以,我们开始就要默认 数组当中有一个 和为 0 的前缀和,即: hash[0] = 1;

完整代码:

class Solution {
public:
    int subarraySum(vector<int>& nums, int k) {
        unordered_map<int, int> hash(nums.size());
        hash[0] = 1;

        int sum = 0, ret = 0; // sum 代替sum数组,利用变量给 hash 当中赋值;ret 返回个数
        for(auto& e : nums)
        {
            sum += e; // 计算当前的前缀和

            // 计算满足 条件的区间,是否在 hash 当中出现。count()函数判断是否出现
            //  出现计数器 加上 这个前缀和 在 hash 当中出现的次数
            if(hash.count(sum - k)) ret += hash[sum - k]; 
            hash[sum]++;                  
        }

        return ret;
    }
};

I974. 和可被 K 整除的子数组 - 力扣(LeetCode)

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

要解决这道题目,首先要知道 同于定理 

也就是 如果 a 和 b 的差值,除上 p ,如果是整除的话,也就是余数是零的话,a 除上 p 的余数 =    b 除上 p 的余数

而且,还需要清楚,在C++ 当中, [负数 % 正数] 的结果是 一个负数

 也就是说,其实[负数 % 正数]   其实结果也是一个 余数,但是这个余数是负数。

所以,针对这种情况,我们要对这个负数进行修正,把他修正为 正数。

所以,当我们在计算 [负数 % 正数] 的 结果之时,其实,计算出的负数的结果在加上 模数(也就是其中的正数),其实就是对应的正数的结果。如下图所示:
 

上述的计算方式对于 a 是负数的情况来说,计算出的结果就是修正之后的结果,也就是修正为 正数之后的结果。

但是上述的 修正方式对于 a 是正数的情况是 计算错误的,因为 对于 a 是正数的情况来说, a % b 本来就是 正确的结果,但是后面又加上了 一个 b,所以是不正确的。

所以,为了 正数和负数统一,我们共用的方式就是  在上述计算式子当中,再 % b 即可。

上述式子就是我们想要的i修正公式了。


所以,按照和这个问题其实就和上述 和为K的子数组求解方式一样了。

先求出所有的 从 0 号数组位置开始的 所以前缀模,保存到一个数组当中,然后 求出 与 K 模的余数即可。

同样优化方式和上述一样,不需要多定义一个 数组来保存 前缀模,这样也不好 查找对应的前缀模,只需要 用一个 hash表来存储即可。

上述问题就被简化为了 在 [0, i - 1] 这个区间当中,找到有多少个前缀和 的 余数等于 (sum %  k + k ) % k。

完整代码:
 

class Solution {
public:
    int subarraysDivByK(vector<int>& nums, int k) {
        unordered_map<int, int> hash;
        hash[0 % k] = 1; // 0 这数的余数

        int sum = 0, ret = 0;
        for(auto& e : nums)
        {
            sum += e;
            int r = (sum%k + k) % k;
            if(hash.count(r)) ret += hash[r];
            hash[r]++;
        }
        return ret;
    }
};

I525. 连续数组 - 力扣(LeetCode)

 给定一个二进制数组 nums , 找到含有相同数量的 0 和 1 的最长连续子数组,并返回该子数组的长度。

示例 1:

输入: nums = [0,1]
输出: 2
说明: [0, 1] 是具有相同数量 0 和 1 的最长连续子数组。
示例 2:

输入: nums = [0,1,0]
输出: 2
说明: [0, 1] (或 [1, 0]) 是具有相同数量0和1的最长连续子数组。

在数组当中,所有的 元素的值要么是 0 ,要么是1。我们需要找到一个 符合要求的最长的连续的子数组,返回这个子数组的长度。

我们在统计个数的时候,其实是可以做到转化的,如果把 0 全部替换为-1,那么我们统计个数的问题,其实就 转化成了 找到一个 元素之和等于 0 的子数组。

所以,这道题目就可以使用前缀和的方法来解决。

使用hash 表来存储 前缀和为 sum 的区间,所以吗,应该是 unordered_map<sum, i > (其中 sum 是前缀和,i 是这个前缀和区间的下标)。

当找到某一个下标,和为  sum,计算出 这个区间的长度,就把这个 对应的 绑定的值存入到哈希表当中。

如果有重复的 sum,但是区间不一样,不需要重新更新原本在 hash 表当中的 <sum , i>, 只需要保留之前存入的 <sum, i> 即可。因为 ,我们需要找到 子区间最长的子数组,那么 下标应该是越考左 ,那么 计算出的 子区间 长度才是最大的。

同样,为了处理特殊情况:当 [0 , i] 这个子区间计算出的前缀和就是0了,那么按照上述  和为K的子数组 这个题目当中逻辑去 找到子区间的话,就会在 -1 为开始的区间去找。

所以,我们需要 在开始 默认 一个子区间的前缀和是0,即  hash[0] = -1;

上述的过程就可以找出所有的 合法的子数组了,现在就是如何计算这个子数组的区间大小?

如上,我们只需要找出 i 和 j 两个下标,使得 [0, i] 的 前缀和 和 [0, j] 的前缀和 相等即可

 所以我们计算出的 区间的 长度就是 i - j

完整代码:

class Solution {
public:
    int findMaxLength(vector<int>& nums) {
        unordered_map<int ,int> hash;
        hash[0] = -1;   // 默认 刚开始 哈希表当中有一个 前缀和为0 的区间

        int sum = 0, ret = 0;
        for(int i = 0 ;i < nums.size(); i++)
        {
            sum += nums[i] == 0 ? -1 : 1; // 如果是 0 就加-1,如是1 就加 1

            // 如果 sum 在hash 当中存在,说明此时就已经找到了 符合条件的子区间
            // 那么就更新的 ret 返回值。
            if(hash.count(sum)) ret = max(ret, i - hash[sum] + 1 - 1);
            else hash[sum] = i;  // sum 在 hash 当中不存在,那么 就添加一个 hash 元素
        }

        return ret;
    }
};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1245631.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

化学气相沉积(CVD)中的TEOS

在半导体制程中&#xff0c;薄膜的沉积是核心的步骤之一&#xff0c;有接触过CVD的小伙伴应该或多或少听过TEOS这种物质&#xff0c;TEOS作为一种重要的沉积源&#xff0c;尤其在低温氧化硅的生成过程中&#xff0c;发挥了无可替代的角色。今天我们就来聊聊这种物质。 什么是TE…

java SpringCloud版本b2b2c鸿鹄云商平台全套解决方案 小程序商城免费搭建

使用技术&#xff1a; Spring CloudSpring BootMybatis微服务服务监控可视化运营 B2B2C平台&#xff1a; 平台管理端(包含自营) 商家平台端(多商户入驻) PC买家端、手机wap/公众号买家端 微服务&#xff08;30个通用微服务如&#xff1a;商品、订单、购物车、个人中心、支…

Microsoft Office 2019下载工具

今天博主继续推出重磅福利——Microsoft Office合集的安装工具。 Microsoft Office是一套由微软公司开发的办公软件&#xff0c;它为 Microsoft Windows 和 Mac OS X而开发。与办公室应用程序一样&#xff0c;它包括联合的服务器和基于互联网的服务。最近版本的 Office 被称为 …

【MySQL】mysql中不推荐使用uuid或者雪花id作为主键的原因以及差异化对比

文章目录 前言什么是UUID?什么是雪花ID?什么是MySql自增ID?优缺点对比UUID:优点1.全球唯一性2.无需数据库支持 缺点1.存储空间大2.索引效率低3.查询效率低 雪花ID&#xff1a;优点1.分布式环境下唯一性 缺点1.依赖于机器时钟2.存储空间较大3.查询效率低 MYSQL自增:优点1.简单…

聚类算法模型的概念、评估及应用

聚类是一种无监督学习方法&#xff0c;其目标是将数据集中的样本分成不同的组别&#xff0c;使得同一组内的样本相似度较高&#xff0c;而不同组之间的样本相似度较低。聚类算法模型通常通过计算样本之间的相似度或距离来实现这一目标。以下是聚类算法模型的概念、评估及应用的…

电脑技巧:推荐八个非常实用的在线网站值得收藏

目录 1、wikihow 干货分享网站 2、次元小镇 二次元必备网站 3、AI创作家 4、SKRbt 搜索引擎网站 5、barbg 全球资源网站 6、书签地球 7、4KHDR世界 8、a real me 今天小编给大家推荐八个非常实用的在线网站值得收藏&#xff01; 1、wikihow 干货分享网站 这个网站是一…

文章解读与仿真程序复现思路——电工技术学报EI\CSCD\北大核心《面向差异化电源成本结构的容量市场机制设计》

这个文章标题涉及到容量市场机制设计&#xff0c;着重考虑了电源成本结构的差异性。下面对标题中的关键词进行解读&#xff1a; 面向&#xff08;Facing&#xff09;&#xff1a; 表示该容量市场机制设计是以某种方向、取向或目标为基础的。在这里&#xff0c;可能指的是设计是…

九州未来联合联通智网科技发布白皮书,促进车联网融合发展

2023年11月21日&#xff0c;由2023中国5G工业互联网大会组委会、工业和信息化部主办&#xff0c;联通智网科技承办的2023中国5G工业互联网大会——5G车联网与智慧交通创新发展平行会议&#xff0c;在武汉成功举办。 九州未来作为中国联通车联网创新联合体成员单位&#xff0c;受…

自学编程,用好这几个网站就够了!

如果你要自学编程&#xff0c;一定要收藏好这7个网站&#xff0c;上面免费的优质教程很多&#xff0c;完全可以省去你上万块钱的学费&#xff01; 话不多说&#xff0c;直接上干货&#xff01; 第一个&#xff0c;W3school 一个主打图文教程的网站&#xff0c;不管是前端开发…

域控操作五:统一熄屏睡眠时间

直接看图路径&#xff0c;我只设置了熄屏&#xff0c;如果要睡眠就下面那个启用设置时间

本地部署 ComfyUI

本地部署 ComfyUI ComfyUI 介绍ComfyUI Github 地址部署 ComfyUI配置模型地址 or 下载模型启动 ComfyUI访问 ComfyUI ComfyUI 介绍 最强大、模块化的稳定扩散 GUI 和后端。 该用户界面将允许您使用基于图形/节点/流程图的界面设计和执行高级稳定扩散管道。 ComfyUI Github 地…

123. 股票买卖的最佳时机III(2次交易)

题目 题解 class Solution:def maxProfit(self, prices: List[int]) -> int:N len(prices)# 状态定义 dp[i][j][k]代表在第i天&#xff0c;被允许完成j次交易时&#xff0c;持有或者不持有的最大利润。k0代表不持有&#xff0c;k1代表持有dp [[[0 for k in range(2)] for…

反爬虫机制与反爬虫技术(二)

反爬虫机制与反爬虫技术二 1、动态页面处理与验证码识别概述2、反爬虫案例:页面登录与滑块验证码处理2.1、用例简介2.2、库(模块)简介2.3、网页分析2.4、Selenium准备操作2.5、页面登录2.6、模糊移动滑块测试3、滑块验证码处理:精确移动滑块3.1、精确移动滑块的原理3.2、滑…

Centos部署GitLab-备份恢复

1. 下载rpm包 wget https://mirrors.tuna.tsinghua.edu.cn/gitlab-ce/yum/el7/gitlab-ce-10.8.4-ce.0.el7.x86_64.rpm2. 安装依赖 yum -y install policycoreutils openssh-server openssh-clients postfix policycoreutils-python3. rpm安装 rpm -ivh gitlab-ce-10.8.4-ce.…

Selenium介绍及基本使用方法

Selenium是一个开源、免费、简单、灵活&#xff0c;对Web浏览器支持良好的自动化测试工具&#xff0c;在UI自动化、爬虫等场景下是十分实用的&#xff0c;能够熟练掌握并使用Selenium工具可以大大的提高效率。 Selenium简介 Selenium支持多平台、多浏览器、多语言去实现自动化…

【开源】基于Vue.js的固始鹅块销售系统

项目编号&#xff1a; S 060 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S060&#xff0c;文末获取源码。} 项目编号&#xff1a;S060&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 数据中心模块2.2 鹅块类型模块2.3 固…

EPT-Net:用于3D医学图像分割的边缘感知转换器

EPT-Net: Edge Perception Transformer for 3D Medical Image Segmentation EPT-Net&#xff1a;用于3D医学图像分割的边缘感知转换器背景贡献实验方法Dual Positional Transformer&#xff08;双位置Transformer&#xff09;Learnable Patch EmbeddingVoxel Spacial Positiona…

【开源】基于Vue.js的用户画像活动推荐系统

项目编号&#xff1a; S 061 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S061&#xff0c;文末获取源码。} 项目编号&#xff1a;S061&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 数据中心模块2.2 兴趣标签模块2.3 活…

[C++历练之路]优先级队列||反向迭代器的模拟实现

W...Y的主页 &#x1f60a; 代码仓库分享&#x1f495; &#x1f354;前言&#xff1a; 在C的宇宙中&#xff0c;优先队列似乎是一座巨大的宝库&#xff0c;藏匿着算法的珍宝。而就在这片代码的天空下&#xff0c;我们不仅可以探索优先队列的神奇&#xff0c;还能够揭开反向迭…

C# Onnx PP-Vehicle 车辆分析(包含:车辆检测,识别车型和车辆颜色)

目录 效果 模型信息 mot_ppyoloe_s_36e_ppvehicle.onnx vehicle_attribute_model.onnx 项目 代码 下载 其他 C# Onnx PP-Vehicle 车辆分析&#xff08;包含&#xff1a;车辆检测&#xff0c;识别车型和车辆颜色&#xff09; 效果 模型信息 mot_ppyoloe_s_36e_ppvehi…