与双指针的亲密接触:快与慢的浪漫交错

在这里插入图片描述

公主请阅

  • 1.合并两个有序数组
    • 1.1 题目说明
      • 示例 1
      • 示例 2
      • 示例 3
    • 1.2 题目分析
    • 1.3代码部分
    • 1.4 代码解析
  • 2.移动零
    • 2.1题目说明
      • 示例 1
      • 示例 2
    • 2.2题目分析
    • 2.3代码部分
    • 2.4代码解析

1.合并两个有序数组

在这里插入图片描述
题目传送门

1.1 题目说明

给你两个按 非递减顺序 排列的整数数组 nums1nums2,另有两个整数 mn,分别表示 nums1nums2 中的元素数目。

请你 合并 nums2nums1 中,使合并后的数组同样按 非递减顺序 排列。

注意
最终,合并后数组不应由函数返回,而是存储在数组 nums1 中。为了应对这种情况,nums1 的初始长度为 m + n,其中前 m 个元素表示应合并的元素,后 n 个元素为 0,应忽略。nums2 的长度为 n


示例 1

输入

  • nums1 = [1, 2, 3, 0, 0, 0]
  • m = 3
  • nums2 = [2, 5, 6]
  • n = 3

输出[1, 2, 2, 3, 5, 6]

解释:需要合并 [1, 2, 3][2, 5, 6]。合并结果是 [1, 2, 2, 3, 5, 6],其中斜体加粗标注的是 nums1 中的元素。


示例 2

输入

  • nums1 = [1]
  • m = 1
  • nums2 = []
  • n = 0

输出[1]

解释:需要合并 [1][]。合并结果是 [1]


示例 3

输入

  • nums1 = [0]
  • m = 0
  • nums2 = [1]
  • n = 1

输出[1]

解释:需要合并的数组是 [][1]。合并结果是 [1]。注意,因为 m = 0,所以 nums1 中没有元素,nums1 中仅存的 0 仅是为了确保合并结果可以顺利存放到 nums1 中。


1.2 题目分析

题目让我们合并两个非递减顺序排列的整数数组nums1nums2,此外还有两个整数mn分别表示nums1nums2中的元素的个数,但是我们的nums1的初始长度是m+nnums2的初始化长度是n,说白了就是直接将nums2中的数据挪到nums1中去,然后在挪动的过程中进行排序的操作

那么对于这种题型,出现了两个数组,那么我们是否能使用双指针算法呢?
那么这里说说我的思路哈:

  • 先定义两个指针,一个指向nums1的有效数据的尾部(即m-1),另外一个指针指向nums2的末尾(即n-1)
  • 我们从nums1的最后一个位置开始进行元素的填充,即从m+n-1这个位置开始
  • 我们依次比较nums1nums2中当前指针指向的元素,将其中较大的元素放到nums1的后面的位置
  • 不断地进行两个指针的移动操作,直到其中一个数组的所有元素都被处理完了
  • 如果nums2中还存在剩余的元素的话,我们直接将nums2中剩下的部分赋值到nums1中,(这个时候的nums1本身的前面就是已经有序了)

那么思路成立,下面就是我们的代码部分了

1.3代码部分


//nums1的初始长度就是m+n
void merge(int* nums1, int nums1Size, int m, int* nums2, int nums2Size, int n)
{int p1=m-1;//nums1最后一个有效元素的下标int p2=n-1;//nums2最后一个有效元素的下标int p=m+n-1;//这个是nums1的总长度的末尾//从后往前进行合并操作while(p1>=0&&p2>=0)//循环停止的条件,只要两个数组中都存在未处理的元素的话,就可以继续进行合并的执行操作{//因为给我们的nums1和nums2都是递减的数组,是有序的//每次比较的时候将两组中大的元素放到p的位置if(nums1[p1]>nums2[p2]){nums1[p]=nums1[p1];p1--;}else{nums1[p]=nums2[p2];p2--;}p--;//放下一个元素,这里是换位置}//到这里的话肯定nums1的指针已经是出界了,不满足循环条件了,但是现在nums2hiatus有元素存在的,那么我们就直接将nums2中剩下的的元素直接复制到nums1中while(p2>=0){nums1[p]=nums2[p2];p2--;p--;}
}

1.4 代码解析

我们先设置好下标为指向位置,就依照我们的这个题的思路那里进行下标设置
然后我们使用这个while循环进行操作
这个while循环的条件是只要我们的两个指针是大于等于0的话就一直进行循环操作,直到出现合并的现象或者是有一个数组的下标变成0了,然后另外一个数组还有元素没有完成迁移

在循环中,我们利用两个下标进行比较的操作
两个数组中元素大的那一个放到nums1的后面
放完之后我们将迁移元素的那个数组的指针往左边移动一位,操作就是指针减减
对于上面的操作存在两种情况,一种是nums1中的那个指针指向的元素大,另一种就是nums2中的元素大了。
进行完上述的操作之后,我们已经让一个元素放到了nums1的末尾了
那么这个末尾的指针就需要向坐进行移动一步的操作了

然后我们出循环
到这里的话就是两种情况,一种就是nums1nums2合并完成了,还有一种就是nums1的指针小于0了,因为nums1里面的元素已经移动完了,那么此时就是nums2中还存在元素没有进行迁移了
那么我们直接将nums2中剩下的元素放到nums1里面就行了,因为此时nums2中的元素都是比nums1小的元素了

2.移动零

在这里插入图片描述

题目传送门

2.1题目说明

给定一个数组 nums,编写一个函数将所有的 0 移动到数组的末尾,同时保持非零元素的相对顺序。

注意
必须在不复制数组的情况下原地对数组进行操作。


示例 1

输入nums = [0, 1, 0, 3, 12]
输出[1, 3, 12, 0, 0]


示例 2

输入nums = [0]
输出[0]


2.2题目分析

这类题可以分为数组划分或者叫做数组分块

解决这类题我们首先就想到了双指针算法

这里的指针是利用数组下标来充当指针

因为在数组中我们可以利用下标索引到对应的元素

我们定义的两个指针一个是dest 一个是cur

两个指针的作用

cur:从左往右扫描数组—遍历数组

dest:已处理的区间内,非0元素的最后一个位置

cur在扫描后会将数组分为两个区间,一个是左边,一个是右边,右边就是待处理的区间,左边就是处理过的

然后再cur处理过的左区间,我们的dest在里面又区分了两个区间,左边的就是非0元素,右边的就是0

所以上面会说dest是已处理的区间内,非0元素的最后一个位置

所以这两个指针将整个数组划分为三个部分

在这里插入图片描述
三个区间:

[0,dest] [dest+1,cur-1] [cur,n-1]

非0元素 0 待处理

一开始我们让cur指向第一个元素,因为一开始我们没有开始扫描没有非0元素,所以让dest先指向-1下标这个元素

我们先让cur进行从左往右的扫描操作

当我们的cur遇到0元素的时候cur直接向后移动一位就行了

如果我们的cur碰到非0元素的话

先让dest++,然后让cur位置和dest位置的元素进行交换的操作

如何做到cur从前往后遍历的过程中

2.3代码部分

class Solution {
public:void moveZeroes(vector<int>& nums){for(int cur=0,dest=-1;cur<nums.size();cur++){if(nums[cur])//当cur指向的数不等于0的话swap(nums[++dest],nums[cur]);//让dest自增1,增完之后正是我想要交换的}}
};

2.4代码解析

如何做到cur从前往后遍历的过程中

1.遇到0元素:cur++

2.遇到非0元素:swap(dest+1,cur)

dest++,cur++

我们在这个函数里面写入一个for循环,然后循环的条件就是直到cur走完这个数组就结束,那么只要这个cur的大小小于这个数组的元素我们就能进行循环操作
通过cur遍历整个数组,只要我们的cur指向的元素不是0的话,我们就进行条件语句,先将dest进行++操作,然后我们将dest指向的元素和cur指向的元素进行交换的操作然后这么0就到后面去了,非0元素就在前面了

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

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

相关文章

汽车免拆诊断案例 | 2022款大众捷达VS5车行驶中挡位偶尔会锁在D3挡

故障现象  一辆2022款大众捷达VS5汽车&#xff0c;搭载EA211发动机和手自一体变速器&#xff0c;累计行驶里程约为4.5万km。该车行驶中挡位偶尔会锁在D3挡&#xff0c;车速最高约50 km/h&#xff0c;且组合仪表上的发动机故障灯和EPC灯异常点亮。 故障诊断  用故障检测仪检…

linux一二三章那些是重点呢

第一章 静态库动态库的区别 什么是库 库文件是计算机上的一类文件&#xff0c;可以简单的把库文件看成一种代码仓库&#xff0c;它提供给使用者一些可以直接 拿来用的变量、函数或类。 如何制作 静态动态库 静态库&#xff1a; GCC 进行链接时&#xff0c;会把静态库中代码打…

MySQL-15.DQL-分页查询

一.DQL-分页查询 -- 分页查询 -- 1. 从 起始索引0 开始查询员工数据&#xff0c;每页展示5条记录 select * from tb_emp limit 0,5; -- 2.查询 第1页 员工数据&#xff0c;每页展示5条记录 select * from tb_emp limit 0,5; -- 3.查询 第2页 员工数据&#xff0c;每页展示5条记…

Golang | Leetcode Golang题解之第491题非递减子序列

题目&#xff1a; 题解&#xff1a; var (temp []intans [][]int )func findSubsequences(nums []int) [][]int {ans [][]int{}dfs(0, math.MinInt32, nums)return ans }func dfs(cur, last int, nums []int) {if cur len(nums) {if len(temp) > 2 {t : make([]int, len(…

4.计算机网络_TCP

可靠与效率 TCP的主要特点&#xff1a; TCP是面向连接的运输层协议&#xff0c;每一条TCP连接只能有两个端点&#xff0c;即&#xff1a;点对点、一对一形式。每一个端口都是一个socket。TCP提供可靠交付的服务TCP提供全双工通信&#xff0c;因为TCP的收发缓冲区是分开的。TC…

java导出带图形的word

先看效果图&#xff1a;方法都是一样的&#xff0c;所以数据只做了前两组 第一步需要准备模版&#xff1a; 新建一个word插入图表&#xff0c;选择想要的图表。 编辑图表&#xff1a;营业额表示数字&#xff0c;季度表示文字。其他的样式编辑可根据自己的需求更改&#xff0c;…

(42)MATLAB中使用fftshift绘制以零为中心的功率谱

文章目录 前言一、MATLAB代码二、仿真结果画图 前言 在分析信号的频率分量时&#xff0c;将零频分量平移到频谱中心会很有帮助。本例给出绘制以零为中心的功率谱的方法。 一、MATLAB代码 代码如下&#xff1a; f 1; % 余弦波的振荡频率&#xf…

400行程序写一个实时操作系统(十):用面向对象思想构建抢占式内核

前言 通过前几章的学习&#xff0c;我们学会了如何为RTOS设计一个合理的内存管理算法。现在&#xff0c;是时候学习设计RTOS内核了。 关于RTOS内核的文章也有很多&#xff0c;但都有一点先射箭再化靶子的意味。要么是代码连篇解释却寥寥无几&#xff0c;要么是要先怎么样再怎么…

大数据linux操作系统

第一关&#xff1a;Linux的初体验 答案&#xff1a; cd / ls -a / &#xff08;里面有空格要注意&#xff09; 第二关&#xff1a;Linux的常用命令 答案&#xff1a; touch newfile mkdir newdir cp newfile newdir/newfileCpy 第三关&#xff1a;Linux查询命令帮助语句…

飞机大战告尾

参考 PPO算法逐行代码详解 链接 通过网盘分享的文件&#xff1a;PlaneWar 链接: https://pan.baidu.com/s/1cbLKTcBxL6Aem3WkyDtPzg?pwd1234 提取码: 1234 10.17关于博客发了又改这件事 悲催的事 今天训练了一早上ppo模型&#xff0c;满怀期待的检测成果时发现一点长进都…

根据语音生成视频33搜帧

33搜帧&#xff0c;是一个能根据语音生成视频的网站&#xff0c;33搜帧 - 视频帧画面搜索引擎 33搜帧是一个使用AI技术构建的视频帧画面搜索引擎&#xff0c;和一般素材平台通过视频标签来搜索视频不同&#xff0c;33搜帧能搜索到视频素材中的每一帧画面&#xff0c;这个功能可…

基于SpringBoot+Vue+uniapp的海产品加工销售一体化管理系统的详细设计和实现(源码+lw+部署文档+讲解等)

详细视频演示 请联系我获取更详细的视频演示 项目运行截图 技术框架 后端采用SpringBoot框架 Spring Boot 是一个用于快速开发基于 Spring 框架的应用程序的开源框架。它采用约定大于配置的理念&#xff0c;提供了一套默认的配置&#xff0c;让开发者可以更专注于业务逻辑而不…

软考(网工)——网络操作系统与应用服务器

文章目录 网络操作系统与应用服务器&#x1f550;本地用户与组1️⃣Windows server 2008R2 本地用户与组2️⃣常见用户组与权限 &#x1f551;活动目录1️⃣活动目录2️⃣活动目录&#xff08;Active Directory&#xff0c;AD)3️⃣活动目录工作组分类 &#x1f552;远程桌面与…

uniapp学习(005-1 详解Part.1)

零基础入门uniapp Vue3组合式API版本到咸虾米壁纸项目实战&#xff0c;开发打包微信小程序、抖音小程序、H5、安卓APP客户端等 总时长 23:40:00 共116P 此文章包含第36p-第p40的内容 文章目录 响应式尺寸单位 rpx各种工具修改ui给的图片的宽度ps操作步骤即时设计操作步骤&…

高级算法设计与分析 学习笔记13 线性规划

注意是线性规划不是动态规划哦 好家伙&#xff0c;这不是凸优化吗&#xff1f; 凸优化标准形式&#xff1a; 先改成统一最大化&#xff08;凸优化那边怎么是统一最小化&#xff1f;&#xff09; 原来的x2正负无所谓&#xff0c;但我希望每个x都是有限制的&#xff0c;所以把它改…

高德开放平台API调用实战指南

本文 一、地图展示1.1 地图初始化与展示1.2 自定义标记 二、路线规划2.1 驾车路线规划2.2 步行路线规划 三、定位服务3.1 使用JavaScript API进行定位3.2 IP定位 四、实时交通信息查询4.1 获取实时交通路况 五、智能调度引擎总结 一、地图展示 地图展示是开发基于地理信息系统…

【MySQL】设置二进制日志文件自动过期,从根源上解决占满磁盘的问题:通过修改 binlog_expire_logs_seconds 配置项

引言 MySQL的二进制日志&#xff08;binlog&#xff09;文件记录了数据库中所有更改的详细信息&#xff0c;包括但不限于对数据的插入、删除、更新&#xff0c;对表和数据库的创建、更改、删除等操作。每一次这样的操作都会在二进制日志中生成一个新的日志事件&#xff0c;并被…

【Linux】命令行下的增删查改之“查看”

致谢:Linux常用命令大全(手册) – 真正好用的Linux命令在线查询网站 提供的命令查询 头部内容获取(head) head命令的功能是显示文件开头的内容&#xff0c;默认值为前10行。 指令参数&#xff1a; -n 定义显示行数 -c 指定显示头部内容的字符数 -v 总是显示文件名的头信…

实现双向链表的增删改查

头文件 #pragma once #define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h> #include <stdlib.h> #include <stdbool.h> typedef int LTDataType; typedef struct ListNode {LTDataType data;struct ListNode* prev;struct ListNode* next; } LTNode; //v…

matlab输入汉字时,输入法在左上角显示解决办法

解决方法&#xff1a; 输入汉字时输入法在左上角显示&#xff08;如图1&#xff09;&#xff0c;将鼠标放在竖着的小点处拖动到工作区合适位置&#xff08;如图2&#xff09;&#xff0c;下次输入汉字时输入法便在图2处显示。 图1 图2