代码随想录二刷 | 数组 | 移除元素

代码随想录二刷 | 数组 | 移除元素

  • 题目描述
  • 解题思路 & 代码实现
    • 暴力解法
    • 双指针法

题目描述

27. 移除元素

给你一个数组 nums 和一个值 val,你需要 原地 移除所有数值等于 val 的元素,并返回移除后数组的新长度。

不要使用额外的数组空间,你必须仅使用 O(1) 额外空间并 原地 修改输入数组。

元素的顺序可以改变。你不需要考虑数组中超出新长度后面的元素。

说明:

为什么返回数值是整数,但输出的答案是数组呢?

请注意,输入数组是以「引用」方式传递的,这意味着在函数里修改输入数组对于调用者是可见的。

你可以想象内部操作如下:

// nums 是以“引用”方式传递的。也就是说,不对实参作任何拷贝
int len = removeElement(nums, val);

// 在函数里修改输入数组对于调用者是可见的。
// 根据你的函数返回的长度, 它会打印出数组中 该长度范围内 的所有元素。
for (int i = 0; i < len; i++) {
print(nums[i]);
}

示例 1:

输入:nums = [3,2,2,3], val = 3
输出:2, nums = [2,2]
解释:函数应该返回新的长度 2, 并且 nums 中的前两个元素均为 2。你不需要考虑数组中超出新长度后面的元素。例如,函数返回的新长度为 2 ,而 nums = [2,2,3,3] 或 nums = [2,2,0,0],也会被视作正确答案。

示例 2:

输入:nums = [0,1,2,2,3,0,4,2], val = 2
输出:5, nums = [0,1,3,0,4]
解释:函数应该返回新的长度 5, 并且 nums 中的前五个元素为 0, 1, 3, 0, 4。注意这五个元素可为任意顺序。你不需要考虑数组中超出新长度后面的元素。

提示:

  • 0 <= nums.length <= 100
  • 0 <= nums[i] <= 50
  • 0 <= val <= 100

解题思路 & 代码实现

暴力解法

使用两层for循环,第一层遍历数组元素,第二层更新数组

class Solution {
public:int removeElement(vector<int> &nums, int val) {int size = nums.size();for (int i = 0; i < size; i++) {if (nums[i] == val) { // 发现需要移除的元素,就将数组集体向前移动一位for (int j = i + 1; j < size; j++) {nums[j - 1] = nums[j];}i--; // 因为下标i以后的数值都向前移动了一位,所以i也向前移动一位size--; // 数组大小也需要-1}}return size;}
};

时间复杂度:O(n^2)
空间复杂度:O(1)

双指针法

通过一个快指针和一个慢指针在一个for循环下完成两个for循环的工作。

  • 快指针:寻找新数组(即不含目标元素的数组)的元素
  • 慢指针:指向更新新数组下标的位置
class Solution {
public:int removeElement(vector<int> &nums, int val) {int slowIndex = 0;int size = nums.size();for (int fastIndex = 0; fastIndex < size; fastIndex++) {if (val != nums[fastIndex]) {nums[slowIndex++] = nums[fastIndex];}}return slowIndex;}
};

相对难理解的是这一句:

nums[slowIndex++] = nums[fastIndex];

它的意思是,当没有找到值为val的元素时(if语句的条件),将fastIndex指向的元素赋给slowIndex指向的元素,随后slowIndex向右移动一位。

它也等同于:

nums[slowIndex] = nums[fastIndex];
slowIndex++;

本题中双指针的目的是在原有数组的基础上重新构建一个不含目标值的新数组,删除操作被巧妙的隐藏在了构建新数组中。

如果一直没遇到目标值,每一次循环过后,两个指针是始终指向同一个元素的;如果遇到了目标值,slow指向的元素依然是非目标值元素,而fast指针会继续向前,在下一次飞目标值的循环时,slow指向的元素会重新获得fast指向的元素,也就是说,目标值没有被slow指向过,他不在新构建的数组中。

举一个例子来帮助理解:

一个数组如下,需要移除的元素是值等于val也就是等于2的元素,初始状态下fastIndex和slowIndex位置如下:
在这里插入图片描述
if (val != nums[fastIndex)的条件下,也就是下标0、1、2这三个位置:

  • fastIndex = 0,nums[fastIndex] = 1,nums[slowIndex] = nums[fastIndex],因此nums[slowIndex] = 1,slowIndex++,此时slowIndex = 1,循环体内结束,最后fastIndex++,fastIndex = 1.
  • fastIndex = 1,nums[fastIndex] = 3,nums[slowIndex] = nums[fastIndex],因此nums[slowIndex] = 3,slowIndex++,此时slowIndex = 2,循环体内结束,最后fastIndex++,fastIndex = 2.
  • fastIndex = 2,nums[fastIndex] = 3,nums[slowIndex] = nums[fastIndex],因此nums[slowIndex] = 3,slowIndex++,此时slowIndex = 3,循环体内结束,最后fastIndex++,fastIndex = 3.

此时的状态如下图:
在这里插入图片描述
此时遇到了目标值,不符合if的条件了,因此fastIndex会直接+1,fastIndex = 4,而slowIndex依然为2:
在这里插入图片描述
接下来的循环中,if条件又满足了,所以:
fastIndex=4,nums[fastIndex] = 1,nums[slowIndex] = nums[fastIndex],所以nums[slowIndex] = 1,slowIndex + 1 = 3。注意此时slowIndex指向了3,而nums[slowIndex] = 4,也就是说,原本的nums[3] = 2,变为了nums[slowIndex] = 1,就相当于删除了原本的nums[3] = 2。此时的数组如下图,可以看到nums[3] = 2 已经被替代了:

在这里插入图片描述
最后fastIndex+1 = 5,slowIndex = 3。接下来全部都符合if的条件,最后fastIndex来到末尾,整个程序结束。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/147836.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【数据结构】F:B DS图_课程表 拓扑排序实现

F : B DS图_课程表 Description 小明这个学期必须选修n门课程&#xff0c;课程编号记为0到n-1。 在选修某些课程之前需要一些先修课程。先修课程按数组prerequisites给出&#xff0c;其中prerequisites[i] [a, b]&#xff0c;表示如果要学习课程a则必须先学习课程b。 例如&a…

Ubuntu/Debian Hat 系 Linux 使用

目录 1. Ubuntu/Debian Hat 系 Linux 使用1.1. 包1.1.1. Install Package1.1.2. Convert .rpm package to .deb1.1.3. Install RPM Package Directly Onto the System on Ubuntu1.1.4. Command not found 1. Ubuntu/Debian Hat 系 Linux 使用 1.1. 包 1.1.1. Install Package…

手搓哈希表、列表、队列,只为了用C语言快速求解华容道游戏,我不是大佬,只是一个游戏算法爱好者

背景 多年前曾经写过C语言求解华容道&#xff0c;当时没有用到哈希表&#xff0c;导致整个查重搜索数组过大&#xff0c;每次求解都得花上数分钟的时间&#xff0c;如今时过境迁&#xff0c;对数据结构和算法有了更深的理解&#xff0c;所以得把这一块补上了。(其实就是最近想…

Redis面经

Redis使用场景 1、缓存&#xff1a; 缓存三兄弟(穿透、击穿、雪崩) 、双写一致、持久化、数据过期策略&#xff0c;数据淘汰策略 2、分布式锁 setnx、redisson 3、消息队列 4、延迟队列 何种数据类型&#xff08;list、zset&#xff09; 缓存三兄弟 缓存穿透 缓存穿透…

Python 如何实现职责链设计模式?什么是职责链设计模式?Python 职责链设计模式示例代码

什么是职责链&#xff08;Chain of Responsibility&#xff09;设计模式&#xff1f; 职责链&#xff08;Chain of Responsibility&#xff09;设计模式是一种行为型设计模式&#xff0c;旨在构建一个对象链&#xff0c;每个对象都有机会处理请求&#xff0c;并且可以将请求传…

动手学深度学习——循环神经网络的从零开始实现(原理解释+代码详解)

文章目录 循环神经网络的从零开始实现1. 独热编码2. 初始化模型参数3. 循环神经网络模型4. 预测5. 梯度裁剪6. 训练 循环神经网络的从零开始实现 从头开始基于循环神经网络实现字符级语言模型。 # 读取数据集 %matplotlib inline import math import torchfrom torch import …

五、hdfs常见权限问题

1、常见问题 2、案例 &#xff08;1&#xff09;问题 &#xff08;2&#xff09;hdfs的超级管理员 &#xff08;3&#xff09;原因 没有使用Hadoop用户对hdfs文件系统进行操作。 在Hadoop文件系统中&#xff0c;Hadoop用户相当于Linux系统中的root用户&#xff0c;是最高级别用…

react实现步进器

创建一个步进器组件&#xff0c;包含当前步骤&#xff08;currentStep&#xff09;的状态以及前进和后退的操作&#xff1a; import React, { useState } from react;function Stepper() {const [currentStep, setCurrentStep] useState(1);const handleNext () > {setCu…

⑩② 【MySQL索引】详解MySQL`索引`:结构、分类、性能分析、设计及使用规则。

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ MySQL索引 ⑩② 【MySQL索引】1. 索引2. 索引的…

搞科研、写论文,如何正确使用GPT?AIGC技术解析、提示词工程高级技巧、AI绘图、ChatGPT/GPT4应用

目录 专题一 OpenAI开发者大会最新技术发展及最新功能应用 专题二 AIGC技术解析 专题三 提示词工程高级技巧 专题四 ChatGPT/GPT4的实用案例 专题五 让ChatGPT/GPT4成为你的论文助手 专题六 让ChatGPT/GPT4成为你的编程助手 专题七 让ChatGPT/GPT4进行数据处理 专题八 …

Wordpress页面生成器:Elementor 插件制作网站页面教程(图文完整)

本文来教大家怎么使用Wordpress Elementor页面编辑器插件来自由创建我们的网页内容。很多同学在面对建站的时候,一开始都是热血沸腾信心满满的,等到实际上手的时候就会发现有很多问题都是无法解决的,希望本篇Elementor插件使用指南能够帮助到你。 Wordpress Elementor页面编…

解决React遍历每次渲染多个根元素导致无法为元素赋值key的问题

遍历时&#xff0c;存在多个根标签&#xff0c;如果使用<></>无法正确赋值key&#xff0c;代码如下&#xff1a; function App() {const list [{ id:1, name:"小明" },{ id:2, name:"小田" },{ id:3, name:"小王" }]const listCon…

使用VC++实现分段线性变换,直方图均衡化、锐化处理(使用拉普拉斯算子)

图像锐化1 获取源工程可访问huiningLi的gitee可在此工程的基础上进行学习。 实验要求 5.1实验目的、要求 实验目的&#xff1a; &#xff08;1&#xff09;掌握图像增强的原理与相关方法。 &#xff08;2&#xff09;能使用VC实现图像增强的一些相关功能。 实验要求&#xf…

【python】均值、中值和高斯滤波详解和示例

本文对均值、中值和高斯滤波进行详解&#xff0c;以帮助大家理解和使用。 这里写目录标题 均值滤波中值滤波高斯滤波核大小为&#xff08;9,9&#xff09;核大小为&#xff08;51,51&#xff09; 小结 下面是示例中使用的原图。 均值滤波 均值滤波是一种简单的平滑滤波器&…

PyTorch微调权威指南3:使用数据增强

如果你曾经参与过 PyTorch 模型的微调&#xff0c;可能会遇到 PyTorch 的内置变换函数&#xff0c;这使得数据增强变得轻而易举。 即使你之前没有使用过这些功能&#xff0c;也不必担心。 在本文中&#xff0c;我们将深入研究 PyTorch 变换换函数的世界。 我们将探索你可以使用…

CTFhub-RCE-过滤目录分隔符 /

根据源代码信息可知&#xff0c;过滤掉了/ <?php $res FALSE; if (isset($_GET[ip]) && $_GET[ip]) { $ip $_GET[ip]; $m []; if (!preg_match_all("/\//", $ip, $m)) { $cmd "ping -c 4 {$ip}"; exec($cmd,…

Java 实现图书管理系统

需求 编写图书借阅程序&#xff0c;可以处理简单的书籍借阅情况&#xff0c;包括借书和还书等。图书类的数据成员包括书名、书号和借书学生等&#xff1b;方法包括借书、还书和显示书籍信息等。学生类的数据成员包括姓名、学号和所借书籍等&#xff1b;方法包括显示学生信息等…

【开源】基于Vue.js的高校实验室管理系统的设计和实现

项目编号&#xff1a; S 015 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S015&#xff0c;文末获取源码。} 项目编号&#xff1a;S015&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、研究内容2.1 实验室类型模块2.2 实验室模块2.3 实…

数据结构与集合源码

我是南城余&#xff01;阿里云开发者平台专家博士证书获得者&#xff01; 欢迎关注我的博客&#xff01;一同成长&#xff01; 一名从事运维开发的worker&#xff0c;记录分享学习。 专注于AI&#xff0c;运维开发&#xff0c;windows Linux 系统领域的分享&#xff01; 本…

【数据结构与算法】线性表 - 顺序表

目录 1. 线性表2.顺序表3.顺序表的优缺点4.实现&#xff08;C语言&#xff09;4.1 头文件 seqList.h4.2 实现 seqList.c 1. 线性表 线性表&#xff08;linear list&#xff09;是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使用的数据结构&#xff0c;常见…