刷爆力扣之构建乘积数组

简介: 刷爆力扣之构建乘积数组

一 🏠 题目描述

剑指 Offer 66. 构建乘积数组


给定一个数组 A[0,1,…,n-1],请构建一个数组 B[0,1,…,n-1],其中 B[i] 的值是数组 A 中除了下标 i 以外的元素的积, 即 B[i]=A[0]×A[1]×…×A[i-1]×A[i+1]×…×A[n-1]。不能使用除法。


示例:


输入: [1,2,3,4,5]
输出: [120,60,40,30,24]

提示:


所有元素乘积之和不会溢出 32 位整数

a.length <=100000


二 🏠破题思路

2.1 🚀 关键信息

解决问题第一步,当然先提取题目字面上的关键信息 😎😎😎


题干简直超易理解,数组 B[0,1,…,n-1],其中 B[i] 的值是数组 A 中除了下标 i 以外的元素的积, 即 B[i]=A[0]×A[1]×…×A[i-1]×A[i+1]×…×A[n-1]



看到这句的时候,各位看官大大应该和博主一样已经笑开了 🌸🌸🌸 吧


但是这道题的核心点是 不能使用除法 😖😖😖



提取完题目中的关键信息后,直接进入第二阶段,思路整理 😃😃😃



2.2 🚀 思路整理

如 2.1 所述,该题难点在于不能使用除法,即 只用乘法 生成数组 B


根据题目对 B[i] 的定义,可列表格,如下图所示


根据表格的主对角线(全为 1 ),可将表格分为 上三角 和 下三角 两部分;分别迭代计算下三角和上三角两部分的乘积,即可 不使用除法 就获得结果




三 🏠 代码详解

3.1 🚀 代码实现

按照我们刚才的破题思路,直接代码走起来 👇👇👇👇


std::vector<int> constructArr(std::vector<int>& a) {
  int len = a.size(); //输入数组长度
  std::vector<int> b(len, 1); //初始化输出数组 b
  int left =1, right =1; //初始化左右双指针
for (int i =0; i < len; ++i) { //遍历输入数组
  b[i] *= left; //left = 索引i位置左边所有数乘积, *= 赋值给 b[i]
  left *= a[i]; //索引i位置左边所有数乘积 * a[i]
  b[len -1- i] *= right; //right = 索引len-1-i位置右边所有数乘积, *= 赋值给 b[len-1-i]
  right *= a[len -1- i]; //索引len-1-i位置右边所有数乘积* a[len -1- i]
  }
  return b;
}


3.2 🚀 细节解析

看完 👀👀👀 全注释版的代码实现后,相信看官大大对整体逻辑已经是大写的 OK 了 😃😃😃


那么我们挖掘上述实现的晦涩细节 😖😖😖 进行解析,直接开干,走起来 👇👇👇👇


b[i] *= left; //left = 索引i位置左边所有数乘积, *= 赋值给 b[i]
left *= a[i]; //索引i位置左边所有数乘积 * a[i]

left 是索引 i 位置左边所有数乘积,*= 就是在把索引 i 位置左边所有数乘积赋值给 b[i]


left *= a[i] 为了一下循环(++i)提前计算 i 位置左边所有数乘积


那么这个过程就对应了,2.2 图示中的正三角过程(右部分的倒三角逻辑同理) 🌹🌹🌹



当然也可以使用两次循环,将正倒三角各元素乘积分开计算,在逻辑上或许会更好理解。但却多了一次遍历的开销,这里使用了双指针的操作(凡是等效逻辑正逆向遍历两次,均可使其简化为一次完成) 😜😜😜

四 🏠 心路历程

为方便各位看官大大了解博主真实刷题过程,我把当时状态纯纯真实还原,记录在心路历程这一小节,不感兴趣的小伙伴可以直接跳过哈


博主在第一阶段提取 🚀 关键信息并没有问题,在第二阶段 🚀 思路整理中有考虑尝试使用双指针,但是未联想到 上三角 和 下三角(破题关键点)😭😭😭


所以博主的这道题是在阅读完官方解析后,解出来并加以记录的


相关文章
|
1月前
|
算法
LeetCode第53题最大子数组和
LeetCode第53题"最大子数组和"的解题方法,利用动态规划思想,通过一次遍历数组,维护到当前元素为止的最大子数组和,有效避免了复杂度更高的暴力解法。
LeetCode第53题最大子数组和
|
1月前
|
存储 Java API
LeetCode------合并两个有序数组(4)【数组】
这篇文章介绍了LeetCode上的"合并两个有序数组"问题,并提供了三种解法:第一种是使用Java的Arrays.sort()方法直接对合并后的数组进行排序;第二种是使用辅助数组和双指针技术进行合并;第三种则是从后向前的双指针方法,避免了使用额外的辅助数组。
LeetCode------合并两个有序数组(4)【数组】
LeetCode------找到所有数组中消失的数字(6)【数组】
这篇文章介绍了LeetCode上的"找到所有数组中消失的数字"问题,提供了一种解法,通过两次遍历来找出所有未在数组中出现的数字:第一次遍历将数组中的每个数字对应位置的值增加数组长度,第二次遍历找出所有未被增加的数字,即缺失的数字。
|
1月前
|
前端开发
LeetCode------移动零(5)【数组】
这篇文章介绍了LeetCode上的"移动零"问题,提出了一种使用双指针的原地操作解法,该方法首先将非零元素移动到数组前端并保持相对顺序,然后填充后续位置为零,以达到题目要求。
|
1月前
|
算法
LeetCode第81题搜索旋转排序数组 II
文章讲解了LeetCode第81题"搜索旋转排序数组 II"的解法,通过二分查找算法并加入去重逻辑来解决在旋转且含有重复元素的数组中搜索特定值的问题。
LeetCode第81题搜索旋转排序数组 II
|
1月前
|
算法 索引
LeetCode第34题在排序数组中查找元素的第一个和最后一个位置
这篇文章介绍了LeetCode第34题"在排序数组中查找元素的第一个和最后一个位置"的解题方法,通过使用双指针法从数组两端向中间同时查找目标值,有效地找到了目标值的首次和最后一次出现的索引位置。
LeetCode第34题在排序数组中查找元素的第一个和最后一个位置
|
1月前
|
算法
LeetCode第33题搜索旋转排序数组
这篇文章介绍了LeetCode第33题"搜索旋转排序数组"的解题方法,通过使用二分查找法并根据数组的有序性质调整搜索范围,实现了时间复杂度为O(log n)的高效搜索算法。
LeetCode第33题搜索旋转排序数组
|
1月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
41 6
|
1月前
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
73 2
|
1月前
|
索引 Python
【Leetcode刷题Python】从列表list中创建一颗二叉树
本文介绍了如何使用Python递归函数从列表中创建二叉树,其中每个节点的左右子节点索引分别是当前节点索引的2倍加1和2倍加2。
36 7