[leetcode]238. 除自身以外数组的乘积
题目描述
给定长度为 n 的整数数组 nums,其中 n > 1,返回输出数组 output ,其中 output[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积。
示例:
输入: [1,2,3,4]
输出: [24,12,8,6]
说明: 请不要使用除法,且在 O(n) 时间复杂度内完成此题。
进阶:
你可以在常数空间复杂度内完成这个题目吗?( 出于对空间复杂度分析的目的,输出数组不被视为额外空间。)
算法
题中明确要求不能使用除法,本可以使用O(n^2)的双for暴力循环求解,但又要求在 O(n) 时间复杂度内。这就需要开动下脑筋,先来分析一下输出是怎么的得到:
24 = 2 * 3* 4
12 = 1 * 3 *4
8 = 1 * 2 * 4
6 = 1 * 2 *3
记保存输出的向量为vec,长度为length。
显然vec[i]上需要填的值 = 前i个数累乘 * 后(length - i - 1)个数累乘
仔细一思考,这个好像可以和动态规划有些关联,毕竟涉及到前i个、后几个连乘这样的关系。现在还处在探索问题的阶段,将问题分为两部分思考,一个是从前往后走,一个是从后往前走,保存它们的累乘结果到2个数组中(先不考虑进阶部分要求的常数空间复杂度),观察这2个数组和我们最终的输出有什么关联。
从前往后
| nums | 1 | 2 | 3 | 4 | 
|---|---|---|---|---|
| dp_forward | 1 | 1 | 2 | 6 | 
if i > 0:
    dp[i] = (nums[0]到nums[i-1]的累乘)
else:
    dp[0] = 1
"""
dp数组的前后值有这么一个数学关系:
                dp[i] = dp[i-1] * nums[i-1]
"""
从后往前
为了方便阅读与理解,将nums反转为{4,3,2,1}。套用上面得到的数学关系,可以很轻松的到下面的结果:
| nums | 4 | 3 | 2 | 1 | 
|---|---|---|---|---|
| dp_backward | 1 | 4 | 12 | 24 | 
将nums反转为正常顺序{1,2,3,4},表格变为:
| nums | 1 | 2 | 3 | 4 | 
|---|---|---|---|---|
| dp_backward | 24 | 12 | 4 | 1 | 
最后一步
我们已经得到从前向后累乘与从后向前累乘的各个位置上的结果,将它们放在一起:
| nums | 1 | 2 | 3 | 4 | 
|---|---|---|---|---|
| dp_forward | 1 | 1 | 2 | 6 | 
| dp_backward | 24 | 12 | 4 | 1 | 
答案呼之欲出,将dp_forward和dp_backward一行对应相乘得到{24,12,8,6}正是需要的输出。
我敢保证这不是偶然,让我们在分析一遍下面这个式子:
vec[i] = 前i个数累乘 * 后(length - i - 1)个数累乘
前i个数累乘正好保存在dp_forward[i]中;同理,后(length - i - 1)个数累乘正好保存在dp_backward[i]中。正因为这样,所以对应相乘,才正好是需要的答案。
进阶部分
在常数空间复杂度内完成这个题目。 出于对空间复杂度分析的目的,输出数组不被视为额外空间。
也就是说,我们最多可以使用一个大小为length的数组外加几个常数这么多的存储空间。
我的想法是从前往后这部分一定要保存在输出数组中,后面从后往前这部分可以直接更新在输出数组中。具体不在赘述,看代码里的注释更好理解。
代码
class Solution {
public:
    vector<int> productExceptSelf(vector<int>& nums) {
        int length = nums.size();
        // 输出数组
        vector<int> vec(length, 1);
        /*** 两次遍历,从前往后一次,从后往前一次 ***/
        // 从前往后
        int mul = 1;
        for (int i = 1; i < length; i++)
        {
            mul *= nums[i-1];
            vec[i] = mul;
        }
        // 从后往前
        mul = 1;
        for (int i = length - 2; i >= 0; i--)
        {
            mul *= nums[i+1];
            // 更新在输出数组上,觉得抽象可以动笔画一画
            vec[i] *= mul;
        }
        return vec;
    }
};
[leetcode]238. 除自身以外数组的乘积的更多相关文章
- LeetCode 238. 除自身以外数组的乘积(Product of Array Except Self)
		238. 除自身以外数组的乘积 238. Product of Array Except Self 题目描述 LeetCode LeetCode238. Product of Array Except ... 
- Java实现 LeetCode 238 除自身以外数组的乘积
		238. 除自身以外数组的乘积 给定长度为 n 的整数数组 nums,其中 n > 1,返回输出数组 output ,其中 output[i] 等于 nums 中除 nums[i] 之外其余各元 ... 
- LeetCode 238. 除自身以外数组的乘积( Product of Array Except Self)
		题目描述 给定长度为 n 的整数数组 nums,其中 n > 1,返回输出数组 output ,其中 output[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积. 示例: 输 ... 
- [LeetCode] 238. 除自身以外数组的乘积 ☆☆☆(左积*右积)
		描述 给定长度为 n 的整数数组 nums,其中 n > 1,返回输出数组 output ,其中 output[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积. 示例: 输入: ... 
- leetcode 238. 除自身以外数组的乘积 (python)
		给定长度为 n 的整数数组 nums,其中 n > 1,返回输出数组 output ,其中 output[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积. 示例: 输入: [1 ... 
- Leetcode题目238.除自身以外数组的乘积(中等)
		题目描述: 给定长度为 n 的整数数组 nums,其中 n > 1,返回输出数组 output ,其中 output[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积. 示例: ... 
- leecode 238除自身以外数组的乘积
		class Solution { public: vector<int> productExceptSelf(vector<int>& nums) { //用除法必须要 ... 
- 剑指offer 66. 构建乘积数组(Leetcode 238. Product of Array Except Self)
		剑指offer 66. 构建乘积数组 题目: 给定一个数组A[0, 1, ..., n-1],请构建一个数组B[0, 1, ..., n-1],其中B中的元素B[i] = A[0] * A[1] * ... 
- LeetCode 81——搜索旋转排序数组 II
		1. 题目 2. 解答 2.1. 方法一 基于 LeetCode 33--搜索旋转排序数组 中的方法二. 当 nums[mid] = nums[right] 时,比如 [1, 1, 2, 1, 1], ... 
随机推荐
- excel查找某一列的值在、不在另一列中
			统计中遇到找出一列的值不在另一列的需求: 找出A列中不在B列的值 方法如下: 使用countif函数 比如找出A列中不在B列的值: 在C1中输入 COUNTIF(B:B,A1) 下拉单元格,在首行添加 ... 
- foreach控件的运用(非原创)http://blog.chinaunix.net/uid-26884465-id-3416869.html
			人们对从认识事物都有一个具体到抽象的过程,学习Jmeter也不例外,通过一个实例来进行学习,一方面可以让初学者有所见即所得的信心,另一方面,其实也是在初学者心中留下了对这事物的一个朦胧的印象,这在以后 ... 
- Tencent interview
			1.常见的聚类算法 1):划分法:k-means 2):基于密度的方法: 2.EM 算法 EM算法是在概率模型中寻找参数的最大似然估计或者最大后验概率的算法,其中概率模型依赖于无法观测的隐藏变量.EM ... 
- vlookup返回多个结果
			http://www.360doc.com/content/12/1021/15/7665211_242782107.shtml =IFERROR(VLOOKUP(D2&ROW(A1),IF( ... 
- [R]关于R语言的绘图函数
			1. 首先就是plot(x,y,...) 参数: x: 所绘图形横坐标构成的对象 y: 所绘图形纵坐标构成的对象 type: 指定所绘图形类型 pch: 指定绘制点时使用的符号 cex: 指定符号的大 ... 
- SpringBoot2 使用Spring Session集群
			有几种办法: 1.扩展指定server利用Servlet容器提供的插件功能,自定义HttpSession的创建和管理策略,并通过配置的方式替换掉默认的策略.缺点:耦合Tomcat/Jetty等Serv ... 
- 【kuangbin专题】计算几何_半平面交
			1.poj3335 Rotating Scoreboard 传送:http://poj.org/problem?id=3335 题意:就是有个球场,球场的形状是个凸多边形,然后观众是坐在多边形的边上的 ... 
- cglib invoke 和 invokeSuper 可用的组合
			在深入字节码理解invokeSuper无限循环的原因中,我们理解的cglib的原理和其中一个合理的调用方式.但是这个调用方式是基于类的,对所有实例生效.实际场景中,我们可能只是希望代理某个具体的实例, ... 
- 【xsy2272】 与运算 状压dp
			题目大意:给你一个长度为$n$的序列$a$,我们定义$f_i$表示序列$a$前i项一次进行按位与运算后的值. 我们认为一个序列的价值为$\sum_{i=1}^{n}f_i$,现在你要重新排列序列$a$ ... 
- WebForm——JS检测浏览器是否是IE浏览器
			function IEVersion() { var userAgent = navigator.userAgent; //取得浏览器的userAgent字符串 && userAgen ... 
