追寻数组的轨迹,解开算法的情愫

在这里插入图片描述

公主请阅

  • 1. 移除元素
    • 1.1 题目说明
      • 示例 1
      • 示例 2
    • 1.2 题目分析
    • 1.3 代码部分
    • 1.4 代码分析
  • 2. 删除有序数组中的重复项
    • 2.1 题目说明
      • 示例 1
      • 示例 3
    • 2.2 题目分析
    • 2.3 代码部分
    • 2.4 代码分析

1. 移除元素

在这里插入图片描述

题目传送门

1.1 题目说明

题目描述
给你一个数组 nums 和一个值 val,你需要原地移除所有数值等于 val 的元素,元素的顺序可能会发生改变。然后返回数组中与 val 不同元素的个数 k

要求:

  1. 修改数组 nums,使得数组的前 k 个元素包含所有与 val 不同的元素。
  2. 数组剩余元素的顺序和大小无关紧要。

函数应返回:数组中 k 的值,即与 val 不同的元素的个数。

示例

示例 1

  • 输入:nums = [3, 2, 2, 3], val = 3
  • 输出:2, nums = [2, 2, _, _]
  • 解释:数组中的前两个元素为 2,移除元素 3

示例 2

  • 输入:nums = [0, 1, 2, 2, 3, 0, 4, 2], val = 2
  • 输出:5, nums = [0, 1, 4, 0, 3, _, _, _]
  • 解释:数组中的前五个元素为 0, 1, 4, 0, 3,移除了 2

提示

  1. 数组的长度范围为:0 <= nums.length <= 100
  2. 数组元素的值范围:0 <= nums[i] <= 50
  3. 要移除的值 val 的范围:0 <= val <= 100

这道题的核心是双指针技巧,用一个指针遍历整个数组,另一个指针记录有效的(与 val 不同的)元素的位置。

1.2 题目分析

这个题让我们将数组中值等于val的进行删除的操作
那么我们就可以使用快慢指针进行操作了
但是这个并非是真正的指针,对于数组来说这个是下标
这个题其实很简单,我们定义快慢指针,快指针进行遍历数组的操作,如果快指针遍历到的数组的元素大小不等于val的话,那么我们就将当前位置赋值给慢指针的位置上面,然后慢指针进行++移动的操作,然后我们就间接的将这个val的值删除了
好的,让我举个具体的例子来说明这个解法。

如果把 i 变为慢指针,而 j 变为快指针,我们的解法和思路会稍有不同,但逻辑依然可以保持不变。具体来说,i 用来记录当前的有效位置,而 j 用来遍历数组。

修改后的思路:

  • 快指针 j 遍历数组的每个元素。
  • 慢指针 i 用来记录下一个不等于 val 的元素要存放的位置。

当我们发现 nums[j] 不等于 val 时,我们将 nums[j] 放到 nums[i] 的位置,然后将 i 向前移动。

具体实现步骤:

  1. 初始化慢指针 i0,表示下一个不等于 val 的元素要存放的位置。
  2. 遍历数组 nums,使用快指针 j
  3. 如果 nums[j] != val,则将 nums[j] 复制到 nums[i],并将慢指针 i 向前移动一位。
  4. 最后,返回 i,此时它表示的是有效数组的长度。

举个例子:

假设输入:

  • nums = [3, 2, 2, 3]
  • val = 3

初始状态:

  • nums = [3, 2, 2, 3]
  • val = 3
  • 慢指针 i = 0,快指针 j = 0

过程演示:

  1. 第一轮

    • j = 0nums[0] = 3,等于 val,跳过。此时,i 不动。
    • 状态:nums = [3, 2, 2, 3]i = 0j = 1
  2. 第二轮

    • j = 1nums[1] = 2,不等于 val,将 nums[1] 复制到 nums[i],即 nums[0] = 2,然后 i++
    • 状态:nums = [2, 2, 2, 3]i = 1j = 2
  3. 第三轮

    • j = 2nums[2] = 2,不等于 val,将 nums[2] 复制到 nums[i],即 nums[1] = 2,然后 i++
    • 状态:nums = [2, 2, 2, 3]i = 2j = 3
  4. 第四轮

    • j = 3nums[3] = 3,等于 val,跳过,i 不动。
    • 状态:nums = [2, 2, 2, 3]i = 2j = 4,遍历结束。

最终结果

  • i = 2,表示新数组的长度为 2,即 nums 的前 2 个元素为 [2, 2]
  • 数组后面的部分内容无关紧要。

解释:
在整个过程中,快指针 j 负责遍历数组,找到不等于 val 的元素后将其复制到慢指针 i 所在的位置,并将慢指针 i 向前移动一位。最终,i 的值就是数组中不等于 val 的元素个数。

总结:
对于 nums = [3, 2, 2, 3],移除 3 后的有效数组长度为 2,修改后的数组前两个元素为 [2, 2]


1.3 代码部分

//使用双指针
//当快指针指向的值不等于要移除的 val 时,将该值赋给慢指针指向的位置,并移动慢指针
int removeElement(int* nums, int numsSize, int val)
{
    int i=0;//慢指针
    for(int j=0;j<numsSize;j++)//快指针遍历整个数组
    {
        if(nums[j]!=val)
        {
            nums[i]=nums[j];//将这个位置的值赋值给慢指针指向的位置
            i++;//慢指针往后移动
        }
    }
    return i;
}

1.4 代码分析

我们先定义变量i0,用来当做慢指针,然后我们在for循环中定义j来当做快指针,我们在for循环中,我们进行判断,如果当前的j下标的值等于我们要删除的val的话,我们直接将这个跳过,但是如果我们遇到不等于val的值的话,我们就将当前的值赋值到i这个位置上面,然后我们的i进行加加的操作,出了循环之后,我们数组中的val已经被删除完了,因为题目让我们返回与val不同的元素,那么我们怎么操作呢?
我们在for循环里面,我们进行了判断,如果不等于val的话,就让当前的值赋值到i的位置上面,并且我们的i也是会走一步的,那么我们i走了多少步,那么就存在多少个不等于val的元素,那么我们将i进行返回就行了


2. 删除有序数组中的重复项

在这里插入图片描述

题目传送门


2.1 题目说明

给你一个 非严格递增排列 的数组 nums,请你 原地 删除重复出现的元素,使每个元素 只出现一次 ,返回删除后数组的新长度。元素的 相对顺序 应该保持一致。然后返回 nums 中唯一元素的个数。

考虑 nums 的唯一元素的数量为 k,你需要做以下事情确保你的题解可以通过:

  • 更改数组 nums,使 nums 的前 k 个元素包含唯一元素,并按照它们最初在 nums 中出现的顺序排列。nums 的其余元素与 nums 的大小无关。
  • 返回 k

判题标准:
系统会用下面的代码来测试你的题解:

int[] nums = [...]; // 输入数组
int[] expectedNums = [...]; // 长度正确的期望答案

int k = removeDuplicates(nums); // 调用

assert k == expectedNums.length;
for (int i = 0; i < k; i++) {
    assert nums[i] == expectedNums[i];
}

如果所有断言都通过,那么您的题解将被 通过

示例 1

输入:nums = [1,1,2]
输出:2, nums = [1,2,_]
解释:函数应该返回新的长度 2,并且原数组 nums 的前两个元素被修改为 [1,2]。不需要考虑数组中超出新长度后面的元素。

示例 3

输入:nums = [0,0,1,1,1,2,2,3,3,4]
输出:5, nums = [0,1,2,3,4,_]
解释:函数应该返回新的长度 5,并且原数组 nums 的前五个元素被修改为 [0,1,2,3,4]。不需要考虑数组中超出新长度后面的元素。

提示:

  • 1 <= nums.length <= 3 * 10^4
  • -10^4 <= nums[i] <= 10^4
  • nums 已按 非严格递增 排列

2.2 题目分析

因为给到我们的数组是有序的,那么这个重复的数字的话肯定是排列在一起的,重复元素是相邻的
我们可以初始化两个指针,i 指向数组的第一个元素,j 从第二个元素开始遍历。如果 nums[j]nums[i] 不相等,说明 nums[j] 是一个新的不同的元素,将其放到 nums[i+1] 的位置上,然后i向前移动一位。那么我们就实现了将重复的第二个元素删除了
然后遍历完数组的话,我们的i+1就是我们新数组的长度了

2.3 代码部分

//只保留每个元素出现的第一个位置的那个元素
//因为这个数组是有序的,所以重复元素肯定是相邻的
//使用一个慢指针 i 来跟踪去重后的数组,并使用一个快指针 j 来遍历数组。
//只要当前元素 nums[j] 不等于 nums[i](即发现了新的元素),就将 nums[j] 放到 i 的下一个位置,并更新 i 的位置。

/*
如果数组为空或者只有一个元素,直接返回数组的长度即可。
初始化两个指针,i 指向数组的第一个元素,j 从第二个元素开始遍历。
如果 nums[j] 和 nums[i] 不相等,说明 nums[j] 是一个新的不同的元素,将其放到 nums[i+1] 的位置上,然后 i 向前移动一位。
遍历完数组后,i + 1 就是新数组的长度。*/
int removeDuplicates(int* nums, int numsSize)
{
    if(numsSize==0)//数组为空,直接返回长度0
    {
        return 0;
    }


    int i =0;
    for(int j=1;j<numsSize;j++)
    {
        if(nums[j]!=nums[i])//发现新元素
        {
            i++;//换下一个元素进行寻找
            nums[i]=nums[j];//将新元素放到重组后的数组中
        }

    }
    //到这里的话就是已经完成了操作重复项了
    return i+1;//返回去重后的数组的长度
}

2.4 代码分析

我们先对特殊情况进行判断,若果当前的数组是空的话,那么我们直接返回0就行了
然后我呢定义一个指针i,初始化为0,然后再定义一个指针j,初始化为1,我们利用j进行遍历数组
如果当前j指向的数字的和我们的i指向的数字是不一样的话,不那么我们就让i往右走一步,然后我们将当前j指向的数字放到我们的i的位置
举个例子:
一个数组:1 1 2
一开始我们的i指向我们的第一个1,j指向我们的第二个1,然后我们进行循环里面的判断操作,因为当前的j下标的元素等于i的指向的元素,那么我们不进行操作,然后我们的j++j指向我们的2,然后我们的2不等于1,所以我们直接先让i走到第二个i个位置,然后我们将j指向的2赋值到当前i的位置,然后我们的数组就变成这样子了 1 2 2
实际的数据只有两个了
那么我们怎么返回删除重复元素的数组的长度呢?
因为我们的i是0开始的,然后只要遇到不同的元素,我们的i就进行加加的操作,那么我们的数组长度就是i+1,那么我们返回i+1就行了,这个就是我们有效数组的长度了

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

代码随想录算法训练营第46期Day42

leetcode.518.零钱兑换 class Solution { public: //求装满背包有几种方法&#xff0c;公式都是&#xff1a;dp[j] dp[j - nums[i]]; // 如果求组合数就是外层for循环遍历物品&#xff0c;内层for遍历背包。 // 如果求排列数就是外层for遍历背包&#xff0c;内层for循环遍历物…

数据结构修炼——常见的排序算法:插入/希尔/选择/堆排/冒泡/快排/归并/计数

目录 一、常见的排序算法二、常见排序算法的实现2.1 排序算法回顾2.1.1 冒泡排序2.1.2 堆排序 2.2 直接插入排序2.3 希尔排序2.4 选择排序2.5 快速排序2.5.1 快速排序&#xff08;霍尔法&#xff09;2.5.2 快速排序&#xff08;挖坑法&#xff09;2.5.3 快速排序&#xff08;前…

Java实现html填充导出pdf

Java实现html填充导出pdf 1.依赖添加和pom修改 <!-- Thymeleaf 模板引擎 --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-thymeleaf</artifactId></dependency><!-- OpenPDF 库 -…

Vue3基于Element-plus的Select组建进行二次封装-demo

效果 组件 <template><ElSelectclass"follow-records-pairs-select"v-model"selectVal":placeholder"placeholder"change"selectChange"><ElOptionv-for"item in options":key"item.value":labe…

点跟踪论文—RAFT: Recurrent All-Pairs Field Transforms for Optical Flow-递归的全对场光流变换

点目标跟踪论文—RAFT: Recurrent All-Pairs Field Transforms for Optical Flow-递归的全对场光流变换 读论文RAFT密集光流跟踪的笔记 RAFT是一种新的光流深度网络结构&#xff0c;由于需要基于点去做目标的跟踪&#xff0c;因此也是阅读了像素级别跟踪的一篇ECCV 2020的经典…

Golang 怎么高效处理ACM模式输入输出

文章目录 问题bufio.NewReader高效的原理 再次提交 问题 最近在练习牛客上单调栈题目时&#xff0c;要求自己处理出入输出&#xff0c;也就是读题库要求的输入&#xff0c;计算最终结果&#xff0c;并打印输出 当我用fmt.Scan处理输入&#xff0c;用fmt.Println处理输出时&am…

使用React和Redux构建可扩展的前端应用

&#x1f496; 博客主页&#xff1a;瑕疵的CSDN主页 &#x1f4bb; Gitee主页&#xff1a;瑕疵的gitee主页 &#x1f680; 文章专栏&#xff1a;《热点资讯》 使用React和Redux构建可扩展的前端应用 1 引言 2 React入门 2.1 安装React 2.2 创建组件 3 Redux基础 3.1 安装Redu…

Jsoup在Java中:解析京东网站数据

对于电商网站如京东来说&#xff0c;其页面上的数据包含了丰富的商业洞察。对于开发者而言&#xff0c;能够从这些网站中提取有价值的信息&#xff0c;进行分析和应用&#xff0c;无疑是一项重要的技能。本文将介绍如何使用Java中的Jsoup库来解析京东网站的数据。 Jsoup简介 …

特殊类设计与设计模式

&#x1f30e;特殊类设计与设计模式 文章目录&#xff1a; 特殊类设计与设计模式 特殊类设计       设计一个只能在堆上创建对象的类       设计一个只能在栈上创建对象的类       请设计一个不能被拷贝的类       请设计一个不能被继承的类 设计模式…

【汇编语言】第一个程序(一)—— 一个源程序从写出到执行的过程

文章目录 前言1. 第一步&#xff1a;编写汇编源程序2. 第二步&#xff1a;对源程序进行编译连接3. 第三步&#xff1a;执行可执行文件中的程序结语 前言 &#x1f4cc; 汇编语言是很多相关课程&#xff08;如数据结构、操作系统、微机原理&#xff09;的重要基础。但仅仅从课程…

【GIT】.cr、.gitattributes 、 .gitignore和.git各文件夹讲解介绍

在 Git 项目中&#xff0c;.cr、.gitattributes 和 .gitignore 文件分别用于不同的配置和管理功能。下面分别解释这些文件的作用和用途&#xff1a; 1. .gitignore 文件 作用&#xff1a; .gitignore 文件用于指定哪些文件或目录应该被 Git 忽略&#xff0c;不会被追踪或提交…

大数据-185 Elasticsearch - ELK 家族 Logstash 安装配置 Input 插件-stdin stdout

点一下关注吧&#xff01;&#xff01;&#xff01;非常感谢&#xff01;&#xff01;持续更新&#xff01;&#xff01;&#xff01; 目前已经更新到了&#xff1a; Hadoop&#xff08;已更完&#xff09;HDFS&#xff08;已更完&#xff09;MapReduce&#xff08;已更完&am…

「C/C++」C++ STL容器库 之 std::string 字符串类

✨博客主页何曾参静谧的博客&#x1f4cc;文章专栏「C/C」C/C程序设计&#x1f4da;全部专栏「VS」Visual Studio「C/C」C/C程序设计「UG/NX」BlockUI集合「Win」Windows程序设计「DSA」数据结构与算法「UG/NX」NX二次开发「QT」QT5程序设计「File」数据文件格式「PK」Parasoli…

vue使用jquery的ajax,页面跳转

一、引入jquery依赖 打开终端更新npm npm install -g npm 更新完后引入输入npm install jquery 加载完后 在最外层的package.json文件中加入以下代码 配置好后导入jquery 设置变量用于接收服务器传输的数据 定义ajax申请数据 服务器的Controller层传输数据 &#xff08;…

linux介绍与基本指令

前言 本次博客将会讲解linux的来源历史、linux操作系统的理解以及它的一些基本指令。 1.linux的介绍 linux的来源 linux的来源最初还是要说到unix操作系统的。 1968年&#xff0c;一些来自通用电器公司、贝尔实验室和麻省理工学院的研究人员开发了一个名叫Multics的特殊操作…

C++ 基于自主实现的红黑树封装Map和Set (下)

C 基于自主实现的红黑树封装Map和Set &#xff08;上&#xff09;-CSDN博客 本文针对上文中没有完成的迭代器接口进行一个补充。 1. 箭头访问 在map的测试中使用箭头访问测试&#xff0c;我们可以复习到: 测试刚才重载的-> , 出现了经典双箭头问题 按理来说应该是像下图一样…

uniapp-components(封装组件)

<myitem></myitem> 在其他类里面这样调用。

Python数值计算(28)——理查森外推法

1. 基础知识 理查森外推法( Richardson extrapolation)是一种提高某些数值过程精度的简单方法&#xff0c;在数值方法中广泛应用。 理查森外推法的基本思想是通过对原函数进行多次求导&#xff0c;并在每一步求导的基础上进行线性组合&#xff0c;得到一个新的函数&#xff0c…

智能时代摩托车一键启动无钥匙进入感受科技前线

向智能化与高性能迈进,技术创新与绿色转型引领摩托车行业智能化出行。 摩托车一键启动无钥匙进入功能是一种先进的车辆控制系统&#xff0c;它允许驾驶员在不使用传统机械钥匙的情况下&#xff0c;通过智能感应技术自动解锁和启动摩托车。这种系统通常包括一个智能钥匙&#x…

从零开始学习 YOLOv8:目标检测与车牌识别实例

1. 引言 什么是目标检测&#xff1f; 目标检测就像是在寻找隐藏的宝藏。想象一下&#xff0c;你在一个巨大的图画里&#xff0c;里面藏着无数的物体&#xff0c;而你的任务是迅速找到其中的几样&#xff0c;比如说&#xff0c;一只流浪的小猫和一辆红色的小轿车。目标检测就是…