§1
题目描述
不使用除法,返回每个位置除自己外其它数的乘积。
nums = [1,2,3,4]
输出 = [24,12,8,6]
§2
思路解析动画文字版
关键不是背模板,而是看懂为什么这题适合前后缀乘积。
1. ans 初始全 1(输入 [1,2,3,4]):求每位「除自己外所有数的乘积」。不用除法 → 答案[i] = 左边所有数之积 × 右边所有数之积。ans 先全置 1。
2. 第一遍 · 填左侧乘积:第一遍从左到右:每位先放左边所有数的乘积。left 滚动累乘:ans = [1, 1, 1×2=2, 1×2×3=6]。
3. 第二遍起 · right=1 从右往左:第二遍从右往左,用一个变量 right(初值 1)记右边乘积。ans[3] ×= right(1) → 6(最右没有右边)。
4. 关键帧 · 再乘右侧乘积:关键帧:继续向左,每位再乘上右边所有数的积(right 也滚动累乘):ans[2]×4=8、ans[1]×12=12、ans[0]×24=24。每位 = 左积 × 右积。
5. 结果:最终 [24,12,8,6]。两遍扫、ans 复用做后缀积 → O(1) 额外空间(不含输出),全程无除法、不怕数组里有 0。
把这句话记住,下次遇到同类题,就能更快选出方向。
迁移阶梯:把这题练到能复述后,再去同类题里迁移:先复用状态定义,再调整边界条件。
边界三连:边界三连:极端输入先过一遍。
几个高频追问,记住答法。
§3
参考代码
class Solution: def productExceptSelf(self, nums): ans = [1] * len(nums) left = 1 for i, x in enumerate(nums): ans[i] = left left *= x right = 1 for i in range(len(nums) - 1, -1, -1): ans[i] *= right right *= nums[i] return ans看懂代码不等于写得出。盖住上面,自己默写一遍试试。卡在哪一行说不清,就点右下角问小欧。
§4
复杂度
- 时间复杂度:O(n),每个核心状态按算法要求处理固定次数
- 空间复杂度:O(1),只保存必要的辅助结构或递归栈
§5
可套用的代码模板
模板不是死背,而是提醒你写代码前先把状态、转移和边界排好。
# 前后缀乘积 通用检查表# 1. 定义状态/指针/容器# 2. 每轮只做一个清晰动作# 3. 更新答案并处理边界§6
易错点
✗ 错误写法:直接把总乘积除以 nums[i]
✓ 正确写法:用左右乘积,完全不做除法
有 0 时除法会崩,而且题目禁止除法
✗ 错误写法:再开一个后缀积数组
✓ 正确写法:用一个滚动变量 right 即可
第二遍从右往左只需一个变量记右积,省掉 O(n) 辅助数组。
✗ 错误写法:忘了 ans 要先初始化
✓ 正确写法:ans 全置 1 再开始累乘
乘法的单位元是 1;初值 0 会让结果全 0。
§
面试追问把动画讲成自己的话
追问为什么不能用除法?
题目明确禁止;且含 0 时除法会除零出错。前后缀积天然避开。
追问怎么做到 O(1) 额外空间?
结果数组先存前缀积,再用一个变量从右往左累乘后缀积写回,不开第二个数组。
追问有 0 怎么办?
前后缀积自动处理:一个 0 时只有它那位非零,两个及以上 0 时全为 0。
这道题到这就讲完了。动画和文字是同一套思路——别光看,关掉页面自己默写一遍。然后顺着主线继续:
下一题 · 数组 & 哈希 8/20
→有效的数独
LeetCode 36 · 中等 · 沿着 数组 & 哈希 继续往下推进
把这道题真正学会,再走
图解算法年卡 ¥99 /年
- ✓本题每步动画的吴师兄语音讲解(全站陆续覆盖)
- ✓小欧带 8 步通关训练——追问到不看答案也能写对、能 30 秒讲给面试官听
- ✓学习报告,记录每道题的掌握程度
299道动画图解
37节语音讲解
28课数据结构
¥0.27折合 / 天
76k+ GitHub Star · 吴师兄开源图解算法,几十万开发者在看的算法讲解
¥99 开通年卡 →
想成体系刷透这类套路?去图解算法专题