LeetCode算法题-Maximum Product of Three Numbers(Java实现)
这是悦乐书的第275次更新,第291篇原创
01 看题和准备
今天介绍的是LeetCode算法题中Easy级别的第143题(顺位题号是628)。给定一个整数数组,从其中找出三个数,使得乘积最大。例如:
输入:[1,2,3]
输出:6
输入:[1,2,3,4]
输出:24
注意:
数组的长度范围为[3,10^4],元素值范围为[-1000,1000]。
任意三个数字的乘积不会超过32位有符号整数的范围。
本次解题使用的开发工具是eclipse,jdk使用的版本是1.8,环境是win7 64位系统,使用Java语言编写和测试。
02 第一种解法
要想三个数的乘积最大化,那么他们的值在数组中肯定是最大的,所以我们需要排序。
元素的取值范围是[-1000,1000],会存在负数的值大于正数的值的情况。对数组排序后,第三个数肯定是数组最后一个元素,如果数组中存在负数,前两个负数的乘积大于倒数第二个元素与倒数第三个元素的乘积,那么第一个数与第二个数就是数组的前两位元素,反之就是数组的倒数第二个元素与倒数第三个元素,我们从这两种情况中取最大值即可。
此解法时间复杂度是O(nlog(n)),空间复杂度是O(1)。
public int maximumProduct(int[] nums) {
if (nums == null || nums.length < 3) {
return 0;
}
// 排序
Arrays.sort(nums);
// 取两种情况的较大者
return Math.max(nums[0]*nums[1]*nums[nums.length-1], nums[nums.length-1]*nums[nums.length-2]*nums[nums.length-3]);
}
03 第二种解法
对第一种解法中借助Arrays.sort方法来实现排序,我们可以再优化下,让排序时间复杂度降到O(n),使用记数排序算法来实现。记数排序算法在之前的一道题目里用过,可以稍后去看看文末的链接。
此解法的时间复杂度是O(n),空间复杂度是O(n),使用了新数组,n为新数组的长度。
public int maximumProduct2(int[] nums) {
if (nums == null || nums.length < 3) {
return 0;
}
// nums的取值范围是[-1000,1000],定义一个新数组的长度比它多一个即可
int[] temp = new int[2001];
for (int n : nums) {
// 以nums的元素值加1000作为新数组的索引,出现次数作为值
temp[n+1000]++;
}
int index = 0;
for (int i=0; i<temp.length; i++) {
// 遇到了nums的元素
if (temp[i] != 0) {
// 可能出现多次,temp[i]为nums中该元素出现的次数
for (int j=0; j<temp[i]; j++) {
nums[index++] = i-1000;
}
}
}
// 依旧取两者中较大的那个
return Math.max(nums[0]*nums[1]*nums[nums.length-1], nums[nums.length-1]*nums[nums.length-2]*nums[nums.length-3]);
}
04 第三种解法
是不是必须要排序才能找到乘积最大的三个元素?
我们可以换个场景,找一组数据中的最大值或者最小值,最简单的做法是直接遍历,依次比较就行,此题我们是不是也可以使用同样的思路?只不过原来是找一个最大值或最小值,现在变成要找三个最大值以及两个最小值了。
此解法的时间复杂度是O(n),空间复杂度是O(1)。
public int maximumProduct3(int[] nums) {
if (nums == null || nums.length < 3) {
return 0;
}
// 定义前三个最大值变量
int max1 = Integer.MIN_VALUE;
int max2 = Integer.MIN_VALUE;
int max3 = Integer.MIN_VALUE;
// 定义前两个最小值变量
int min1 = Integer.MAX_VALUE;
int min2 = Integer.MAX_VALUE;
for (int n : nums) {
// 做比较并进行替换,从后往前
if (n > max1) {
max3 = max2;
max2 = max1;
max1 = n;
} else if (n > max2) {
max3 = max2;
max2 = n;
} else if (n > max3) {
max3 = n;
}
if (n < min1) {
min2 = min1;
min1 = n;
} else if (n < min2) {
min2 = n;
}
}
// 同样是取两者之间的较大者
return Math.max(max1*max2*max3, min1*min2*max1);
}
05 小结
此题本质上是找数组中的最大值或最小值,无论是找单个还是多个,无论是使用排序算法还是其他方式,核心思路都不变。至于上面第二种解法中的记数排序算法,有个使用前提,就是得知道数组元素的取值范围,如果题目有给取值范围是最好的,没给的话,就需要自己去取了。
算法专题目前已日更超过四个月,算法题文章143+篇,公众号对话框回复【数据结构与算法】、【算法】、【数据结构】中的任一关键词,获取系列文章合集。
以上就是全部内容,如果大家有什么好的解法思路、建议或者其他问题,可以下方留言交流,好看、留言、转发就是对我最大的回报和支持!
LeetCode算法题-Maximum Product of Three Numbers(Java实现)的更多相关文章
- LeetCode算法题-Maximum Average Subarray I(Java实现)
这是悦乐书的第278次更新,第294篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第146题(顺位题号是643).给定由n个整数组成的数组,找到具有最大平均值的长度为k的 ...
- LeetCode算法题-Maximum Depth of N-ary Tree(Java实现)
这是悦乐书的第261次更新,第274篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第128题(顺位题号是559).给定n-ary树,找到它的最大深度.最大深度是从根节点到 ...
- LeetCode算法题-Maximum Depth of Binary Tree
这是悦乐书的第164次更新,第166篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第23题(顺位题号是104).给定二叉树,找到它的最大深度.最大深度是从根节点到最远叶节 ...
- 【算法】LeetCode算法题-Maximum Subarray
这是悦乐书的第154次更新,第156篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第13题(顺位题号是53).给定一个整数数组nums,找出一个最大和,此和是由数组中索引 ...
- LeetCode算法题-Unique Morse Code Words(Java实现)
这是悦乐书的第318次更新,第339篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第186题(顺位题号是804).国际莫尔斯电码定义了一种标准编码,其中每个字母映射到一系 ...
- LeetCode算法题-Robot Return to Origin(Java实现)
这是悦乐书的第281次更新,第298篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第149题(顺位题号是657).在2D平面上有一个从位置(0,0)开始的机器人.给定其移 ...
- LeetCode算法题-Min Cost Climbing Stairs(Java实现)
这是悦乐书的第307次更新,第327篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第176题(顺位题号是746).在楼梯上,第i步有一些非负成本成本[i]分配(0索引). ...
- LeetCode算法题-Longest Word in Dictionary(Java实现)
这是悦乐书的第303次更新,第322篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第171题(顺位题号是720).给出表示英语词典的字符串单词数组,找到单词中长度最长的单 ...
- LeetCode算法题-1-bit and 2-bit Characters(Java实现)
这是悦乐书的第302次更新,第321篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第170题(顺位题号是717).有两个特殊字符,第一个字符可以用一个比特0表示,第二个字 ...
随机推荐
- 3.django Model
django ORM基本配置 django中遵循 Code Frist 的原则,即:根据代码中定义的类来自动生成数据库表 1.修改project数据库配置 (1)settigs.py里面 默认 DAT ...
- 『2019/3/8 USACO测试 反思与总结』
2019/3/8 USACO测试 这一次是到高中的第一次考试,考得不太好,原因有很多. 先看一下试题安排: 题号 试题分组 考察算法 思维难度 代码难度 1 金组\(T1\) 建图+最短路 ★★★ ★ ...
- Dubbo(一) —— 基础知识和项目搭建
一.分布式基础理论 1.什么是分布式系统? <分布式系统原理与范型>定义: “分布式系统是若干独立计算机的集合,这些计算机对于用户来说就像单个相关系统” 分布式系统(distribut ...
- asp.net core AuthenticationMiddleware 在WebApi中的的使用
在.net framework 4.5架构下使用认证(Authentication)授权(Authorization). IIS使用HttpModule进行认证(Authentication),我们可 ...
- node 调试相关
#0 node 正确的书写方式 为了防止后面出现混乱的各种书写,先来了解一下如何正确书写 node 的名称. 下面使用来自@bitandbang 推文中的图片展示如何正确书写 node 名称. nod ...
- JavaScript面向对象--多态
一.多态的概念 相同的函数作用于不同的对象,会得到不同的结果,这就是多态. 二.如果不用多态,会怎么样? 这里有个浅显易懂的例子,定义一个函数叫makeSound,传入不同的对象,函数体里要写不同的情 ...
- Android事件机制之二:onTouch详解
<Android事件机制之一:事件传递和消费>一文总结了Android中的事件传递和消费机制. 在其中对OntachEvent中的总结中,不是很具体.本文将主要对onTach进行总结. o ...
- 网络协议抓包分析——TCP传输控制协议(连接建立、释放)
前言 TCP协议为数据提供可靠的端到端的传输,处理数据的顺序和错误恢复,保证数据能够到达其应到达的地方.TCP协议是面向连接的,在两台主机使用TCP协议进行通信之前,会先建立一个TCP连接(三次握手) ...
- javascript基础修炼(2)——What's this(上)
目录 一.this是什么 二.近距离看this 三. this的一般指向规则 四. 基本规则示例 五. 后记 开发者的javascript造诣取决于对[动态]和[异步]这两个词的理解水平. 一.thi ...
- MySQL 笔记整理(8.b) --事务到底是隔离还是不隔离的?
笔记记录自林晓斌(丁奇)老师的<MySQL实战45讲> (本篇内图片均来自丁奇老师的讲解,如有侵权,请联系我删除) 8.a) --事务到底是隔离还是不隔离的? 本周工作较忙,加上懒惰,拖更 ...