66. 构建乘积数组
comments: true
difficulty: 简单
edit_url: https://github.com/doocs/leetcode/edit/main/lcof/%E9%9D%A2%E8%AF%95%E9%A2%9866.%20%E6%9E%84%E5%BB%BA%E4%B9%98%E7%A7%AF%E6%95%B0%E7%BB%84/README.md
面试题 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
解法
方法一:两次遍历
我们先创建一个长度为 n n n 的答案数组 a n s ans ans。
接下来,我们从左到右遍历数组 a a a,过程中维护一个变量 l e f t left left,表示当前元素左边所有元素的乘积,初始时 l e f t = 1 left=1 left=1。当遍历到 a [ i ] a[i] a[i] 时,我们将 l e f t left left 赋值给 a n s [ i ] ans[i] ans[i],然后 l e f t left left 乘以 a [ i ] a[i] a[i],即 l e f t ← l e f t × a [ i ] left \leftarrow left \times a[i] left←left×a[i]。
然后,我们从右到左遍历数组 a a a,过程中维护一个变量 r i g h t right right,表示当前元素右边所有元素的乘积,初始时 r i g h t = 1 right=1 right=1。当遍历到 a [ i ] a[i] a[i] 时,我们将 a n s [ i ] ans[i] ans[i] 乘上 r i g h t right right,然后 r i g h t right right 乘以 a [ i ] a[i] a[i],即 r i g h t ← r i g h t × a [ i ] right \leftarrow right \times a[i] right←right×a[i]。
最终,数组 a n s ans ans 即为所求的答案。
时间复杂度 O ( n ) O(n) O(n),其中 n n n 为数组 a a a 的长度。忽略答案数组的空间消耗,空间复杂度 O ( 1 ) O(1) O(1)。
Python3
class Solution:def constructArr(self, a: List[int]) -> List[int]:n = len(a)ans = [0] * n # 记录乘积数组 left = right = 1for i in range(n):ans[i] = left # ans[0]=1,先累乘i左边乘积left *= a[i]for i in range(n - 1, -1, -1):ans[i] *= right # 再累成i右边乘积right *= a[i]return ans
Java
class Solution {public int[] constructArr(int[] a) {int n = a.length;int[] ans = new int[n];for (int i = 0, left = 1; i < n; ++i) {ans[i] = left; left *= a[i];}for (int i = n - 1, right = 1; i >= 0; --i) {ans[i] *= right;right *= a[i];}return ans;}
}
C++
class Solution {
public:vector<int> constructArr(vector<int>& a) {int n = a.size();vector<int> ans(n);for (int i = 0, left = 1; i < n; ++i) {ans[i] = left;left *= a[i];}for (int i = n - 1, right = 1; ~i; --i) {ans[i] *= right;right *= a[i];}return ans;}
};
Go
func constructArr(a []int) []int {n := len(a)ans := make([]int, n)for i, left := 0, 1; i < n; i++ {ans[i] = leftleft *= a[i]}for i, right := n-1, 1; i >= 0; i-- {ans[i] *= rightright *= a[i]}return ans
}
TypeScript
function constructArr(a: number[]): number[] {const n = a.length;const ans: number[] = Array(n);for (let i = 0, left = 1; i < n; ++i) {ans[i] = left;left *= a[i];}for (let i = n - 1, right = 1; i >= 0; --i) {ans[i] *= right;right *= a[i];}return ans;
}
JavaScript
/*** @param {number[]} a* @return {number[]}*/
var constructArr = function (a) {const n = a.length;const ans = new Array(n);for (let i = 0, left = 1; i < n; ++i) {ans[i] = left;left *= a[i];}for (let i = n - 1, right = 1; ~i; --i) {ans[i] *= right;right *= a[i];}return ans;
};
C#
public class Solution {public int[] ConstructArr(int[] a) {int n = a.Length;int[] ans = new int[n];int left = 1, right = 1;for (int i = 0; i < n; i++) {ans[i] = left;left *= a[i];}for (int i = n - 1; i > -1; i--) {ans[i] *= right;right *= a[i];}return ans;}
}
Swift
class Solution {func constructArr(_ a: [Int]) -> [Int] {let n = a.countguard n > 0 else { return [] }var ans = [Int](repeating: 1, count: n)var left = 1for i in 0..<n {ans[i] = leftleft *= a[i]}var right = 1for i in (0..<n).reversed() {ans[i] *= rightright *= a[i]}return ans}
}