【LeetCode Hot100】除自身以外数组的乘积|左右乘积列表,Java实现!图解+代码,小白也能秒懂!

💻 [LeetCode Hot100] 除自身以外数组的乘积|左右乘积列表,Java实现!图解+代码,小白也能秒懂!

✏️本文对应题目链接:除自身以外数组的乘积


📌 题目描述

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

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

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

示例:

java 复制代码
输入:nums = [1,2,3,4]
输出:[24,12,8,6]

🧠 解题思路(图文分解)

❗ 核心难点

如何在O(n)时间复杂度内,不使用除法,计算每个元素除自身以外的乘积?


方法一:左右乘积列表(黄金思路)✨

关键步骤:

  1. 初始化
    • 创建两个数组 leftright,分别存储每个元素左侧和右侧的乘积
  2. 计算左侧乘积
    • 从左到右遍历数组,累乘左侧元素
  3. 计算右侧乘积
    • 从右到左遍历数组,累乘右侧元素
  4. 计算结果
    • 对于每个元素,answer[i] = left[i] * right[i]

图解左右乘积列表

输入数组:

复制代码
nums = [1, 2, 3, 4]

步骤1:计算左侧乘积

复制代码
left[0] = 1
left[1] = 1 * 1 = 1
left[2] = 1 * 2 = 2
left[3] = 2 * 3 = 6
left = [1, 1, 2, 6]

步骤2:计算右侧乘积

复制代码
right[3] = 1
right[2] = 1 * 4 = 4
right[1] = 4 * 3 = 12
right[0] = 12 * 2 = 24
right = [24, 12, 4, 1]

步骤3:计算结果

复制代码
answer[0] = left[0] * right[0] = 1 * 24 = 24
answer[1] = left[1] * right[1] = 1 * 12 = 12
answer[2] = left[2] * right[2] = 2 * 4 = 8
answer[3] = left[3] * right[3] = 6 * 1 = 6
answer = [24, 12, 8, 6]

🚀 代码实现

java 复制代码
class Solution {
    public int[] productExceptSelf(int[] nums) {
        int n = nums.length;
        int[] answer = new int[n];

        // 计算左侧乘积
        int[] left = new int[n];
        left[0] = 1;
        for (int i = 1; i < n; i++) {
            left[i] = left[i - 1] * nums[i - 1];
        }

        // 计算右侧乘积
        int[] right = new int[n];
        right[n - 1] = 1;
        for (int i = n - 2; i >= 0; i--) {
            right[i] = right[i + 1] * nums[i + 1];
        }

        // 计算结果
        for (int i = 0; i < n; i++) {
            answer[i] = left[i] * right[i];
        }

        return answer;
    }
}

💡 复杂度分析

  • 时间复杂度:O(n) → 遍历数组三次
  • 空间复杂度 :O(n) → 需要两个辅助数组 leftright

方法二:空间优化(进阶思路)

关键思路 :用输出数组 answer 代替左侧乘积列表,使用变量动态计算右侧乘积。

java 复制代码
class Solution {
    public int[] productExceptSelf(int[] nums) {
        int n = nums.length;
        int[] answer = new int[n];

        // 计算左侧乘积
        answer[0] = 1;
        for (int i = 1; i < n; i++) {
            answer[i] = answer[i - 1] * nums[i - 1];
        }

        // 计算右侧乘积并更新结果
        int right = 1;
        for (int i = n - 1; i >= 0; i--) {
            answer[i] = answer[i] * right;
            right = right * nums[i];
        }

        return answer;
    }
}

🌟 总结要点

左右乘积核心思想 :将问题分解为左侧乘积和右侧乘积

空间优化技巧 :用输出数组代替额外空间

适用场景:数组累积问题、前缀后缀问题


相关推荐
IT小码哥丶1 分钟前
华为仓颉语言初识:并发编程之同步机制(上)
java·开发语言
Java技术小馆2 分钟前
打印高质量日志的10条军规
java·后端·面试
fen_fen12 分钟前
学习笔记(26):线性代数-张量的降维求和,简单示例
笔记·学习·算法
王禄DUT13 分钟前
炉石传说 第八次CCF-CSP计算机软件能力认证
c++·算法
白熊18839 分钟前
【推荐算法】DeepFM:特征交叉建模的革命性架构
算法·架构·推荐算法
L_cl41 分钟前
【Python 算法零基础 4.排序 ⑪ 十大排序算法总结】
python·算法·排序算法
小刘不想改BUG1 小时前
LeetCode 70 爬楼梯(Java)
java·算法·leetcode
老歌老听老掉牙1 小时前
使用 SymPy 进行向量和矩阵的高级操作
python·线性代数·算法·矩阵·sympy
张伯毅1 小时前
Java 类型参数 T、R 、 O 、K、V 、E 、? 区别
java·开发语言
lifallen1 小时前
Flink checkpoint
java·大数据·算法·flink