LeetCode Day 13
给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度。
不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的条件下完成。
思路:
因为是有序数组,设置一个max,当前走到哪个值了,后续的值比它小或者等于它,那都是重复值,抛弃该位即可。
/**
* @param {number[]} nums
* @return {number}
*/
var removeDuplicates = function (nums) {
let max = Number.NEGATIVE_INFINITY;
let cursor = 0;
while (cursor < nums.length) {
if (nums[cursor] <= max) {
//重复了,抛弃它
nums.splice(cursor, 1);
}
else {
//新值
max = nums[cursor];
cursor++;
}
}
return nums.length;
};
执行用时 :88 ms,在所有 JavaScript 提交中击败了65.71%的用户
内存消耗 :38 MB,在所有 JavaScript 提交中击败了8.44%的用户
似乎空间消耗的还是比较大,大概是因为splice每次都创建了一个返回数组,虽然我这里没有任何变量引用它,想减少空间的使用量,那我们只好考虑出现重复值的时候,在找到一个不重复的数值之前,把后续的数组往前移动若干位置覆盖它即可。
/**
* @param {number[]} nums
* @return {number}
*/
var removeDuplicates = function (nums) {
if (!nums || nums.length === 0) return 0;
let validCount = nums.length;
let duplicate = 0;
for (let i = 0; i < validCount; i++) {
while (i + 1 + duplicate < validCount && nums[i] === nums[i + 1 + duplicate]) {
duplicate++;
}
if (duplicate > 0) {
//将后续的数据往前移动
for (let j = i + 1 + duplicate; j < validCount; j++) {
nums[j - duplicate] = nums[j];
}
}
validCount -= duplicate;
duplicate = 0;
}
return validCount;
};
执行用时 :188 ms,在所有 JavaScript 提交中击败了14.89%的用户
内存消耗 :37.2 MB,在所有 JavaScript 提交中击败了38.56%的用户
只引入了两个常数变量,空间复杂度没什么变化,数组的移动上导致几乎多了一倍有余的时间,而空间仅仅节省了0.8MB,完全得不偿失。
我们试试官方题解:
/**
* @param {number[]} nums
* @return {number}
*/
var removeDuplicates = function (nums) {
if (!nums || nums.length === 0) return 0;
if (nums.length == 0) return 0;
let i = 0;
for (let j = 1, lens = nums.length; j < lens; j++) {
if (nums[j] != nums[i]) {
i++;
nums[i] = nums[j];
}
}
return i + 1;
};
执行用时 :64 ms, 在所有 JavaScript 提交中击败了99.92%的用户
内存消耗 :37.2 MB,在所有 JavaScript 提交中击败了38.25%的用户
这个做法的关键点在于,j一直往后找,找到一个不跟当前重复的值,就直接插入到当前位置之后,其实它只是找下一个比当前大的值,插入到前面的值之后而已,而不是后续的整体部分集体往前移动。
给定一个数组 nums 和一个值 val,你需要原地移除所有数值等于 val 的元素,返回移除后数组的新长度。
不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的条件下完成。
元素的顺序可以改变。你不需要考虑数组中超出新长度后面的元素。
示例:
给定 nums = [0,1,2,2,3,0,4,2], val = 2,
函数应该返回新的长度 5, 并且 nums 中的前五个元素为 0, 1, 3, 0, 4。
注意这五个元素可为任意顺序。
你不需要考虑数组中超出新长度后面的元素。
参考0026这道题的官方解法,我们很容易可以得出以下的代码:
/**
* @param {number[]} nums
* @param {number} val
* @return {number}
*/
var removeElement = function (nums, val) {
let cur = -1;
for (let next = 0, lens = nums.length; next < lens; next++) {
if (nums[next] !== val) {
cur++;
nums[cur] = nums[next];
}
}
return cur + 1;
};
执行用时 :72 ms, 在所有 JavaScript 提交中击败了33.58%的用户
内存消耗 :34.2 MB, 在所有 JavaScript 提交中击败了5.04%的用户
空间的话只是两个常数指针变量差不了多少,但是时间确实可以优化一下。
考虑到nums=[5,1,2,3,4],然后val=5的情况,上面的写法就需要将后续的1,2,3,4都依次往前面赋值1次,一共赋值4次。
因为题目不要求数组的顺序,如果我们发现某个值不满足需要,我们可以直接把它跟数组最后一个进行交换,然后将有效的数组长度减少1,这样可以减少数字的移动次数和赋值次数。
/**
* @param {number[]} nums
* @param {number} val
* @return {number}
*/
var removeElement = function (nums, val) {
let current = 0;
let validLens = nums.length;
while (current < validLens) {
if (nums[current] === val) {
nums[current] = nums[validLens - 1];
validLens--;
} else {
current++;
}
}
return current;
};
执行用时 :64 ms,在所有 JavaScript 提交中击败了76.40%的用户
内存消耗 :34.2 MB,在所有 JavaScript 提交中击败了5.04%的用户
LeetCode Day 13的更多相关文章
- 【LeetCode算法-13】Roman to Integer
LeetCode第13题 Roman numerals are represented by seven different symbols: I, V, X, L, C, D and M. Symb ...
- 【一天一道LeetCode】#13. Roman to Integer
一天一道LeetCode系列 (一)题目 Given a roman numeral, convert it to an integer. Input is guaranteed to be with ...
- [LeetCode] 12,13 整数和罗马数互转
12. 整数转罗马数字 题目链接:https://leetcode-cn.com/problems/integer-to-roman/ 题目描述: 罗马数字包含以下七种字符: I, V, X, L,C ...
- C# 写 LeetCode easy #13 Roman to Integer
13.Roman to Integer Roman numerals are represented by seven different symbols: I, V, X, L, C, D and ...
- LeetCode题解(13)--Roman to Integer
https://leetcode.com/problems/roman-to-integer/ 原题: Given a roman numeral, convert it to an integer. ...
- python刷LeetCode:13. 罗马数字转整数
难度等级:简单 题目描述: 罗马数字包含以下七种字符: I, V, X, L,C,D 和 M. 字符 数值I 1V 5X 10L 50C 100D 500M 1000例如, 罗马数字 2 写做 II ...
- leetcode算法13.罗马数字转整数
哈喽!大家好,我是[学无止境小奇],一位热爱分享各种技术的博主! [学无止境小奇]的创作宗旨:每一条命令都亲自执行过,每一行代码都实际运行过,每一种方法都真实实践过,每一篇文章都良心制作过. [学无止 ...
- 算法练习之leetcode系列1-3
1.Reverse Words in a String public class Solution { public String reverseWords(String s) { String re ...
- 【LeetCode】13. Roman to Integer 罗马数字转整数
题目: Given a roman numeral, convert it to an integer. Input is guaranteed to be within the range from ...
- 【leetcode】13. Roman to Integer
题目描述: Given a roman numeral, convert it to an integer. 解题分析: 这道题只要百度一下转换的规则,然后着这解释写代码即可.实现上并没有什么难度,直 ...
随机推荐
- JavaScript—飞机大战2版
前面的思路对了 BUG 出在了计时器和没有加判断页面是否存在元素 <!DOCTYPE html> <html lang="en"> <head> ...
- leetcode 690.员工的重要性
题目: 给定一个保存员工信息的数据结构,它包含了员工唯一的id,重要度 和 直系下属的id. 比如,员工1是员工2的领导,员工2是员工3的领导.他们相应的重要度为15, 10, 5.那么员工1的数据结 ...
- JDK的安装与环境变量配置
1.下载JDK后安装,此处安装的是JDK8 2.安装后的路径如下图所示,JDK与JRE在同一个文件夹中 3.安装完JDK后配置环境变量 计算机→属性→高级系统设置→高级→环境变量 4.系统变量→新建 ...
- 计蒜客 密码锁(BFS)
https://www.jisuanke.com/course/1797/121114 Description 现在一个紧急的任务是打开一个密码锁.密码由四位数字组成,每个数字从 1 到 9 进行编号 ...
- 实操windows2008搭建IIS php mysql
一.IIS的安装直接略过 二.主要记录PHP.MYSQL环境的搭建 1.本次环境搭建使用的环境版本号对应如下: 1.PHP:PHP 7.2 (7.2.28) 下载地址:https://windows. ...
- python3.x设置默认编码(sys.stdout.encoding和sys.defaultencoding)
查了一会资料得出的结论是如果你用的是python3.x,那么就最好别去设置sys.defaultencoding或者sys.stdout.encoding记住在需要编码的时候用encode,解码的时候 ...
- pytorch(ch5
读取图片数据集::# -*- coding: utf-8 -*-import torch as tfrom torch.utils import dataimport osfrom PIL impor ...
- Redis--初识Redis
Redis 是一个远程内存数据库,它不仅性能强劲,而且还具有复制特性以及为解决问题而生的独一无二的数据模型.Redis 提供了 5 种不同类型的数据结构,各式各样的问题都可以很自然的映射到这些数据结构 ...
- quartz定时任务(数据库需要的表)
Quartz将Job保存在数据库中所需表的说明 QRTZ_CALENDARS 以 Blob 类型存储 Quartz 的 Calendar 信息 QRTZ_CRON_TRIGGERS 存储 Cron T ...
- cygwin下命令行下切换目录
比我们正常切换目录多个挂载的文件夹 cygdrive