面试经典算法系列之双指针1 -- 合并两个有序数组

面试经典算法题1 – 合并两个有序数组

LeetCode.88
公众号:阿Q技术站

问题描述

给你两个按 非递减顺序 排列的整数数组 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 中。

提示:

  • nums1.length == m + n
  • nums2.length == n
  • 0 <= m, n <= 200
  • 1 <= m + n <= 200
  • -109 <= nums1[i], nums2[j] <= 109

思路

这里给大家讲解一下经典的双指针法

  1. 初始化两个指针 p1p2 分别指向 nums1nums2 的末尾(即初始位置为 m - 1n - 1)。
  2. 初始化一个指针 p,指向 nums1 数组的末尾(即初始位置为 m + n - 1)。
  3. 从后向前遍历 nums1nums2数组,比较nums1[p1]nums2[p2]的大小。
    • 如果 nums1[p1] >= nums2[p2],将 nums1[p] 设为 nums1[p1],并将 p1p 向前移动一位;
    • 如果 nums1[p1] < nums2[p2],将 nums1[p] 设为 nums2[p2],并将 p2p 向前移动一位;
  4. 重复步骤 3 直到 p1p2 小于 0。

这里给大家也演示一下合并过程:

初始状态:

p1=2,p2=2,p=5;

nums1[p1] = 3, num2[p2] = 6

image-20240117201439469

第一步:

nums1[p1] < nums2[p2]nums1[p--] = nums2[p2--]

image-20240117201507836

第二步:

nums1[p1] < nums2[p2]nums1[p--] = nums2[p2--]

image-20240117201532408

第三步:

nums1[p1] >= nums2[p2]nums1[p--] = nums1[p1--]

image-20240117201623941

第四步:

nums1[p1] >= nums2[p2]nums1[p--] = nums1[p1--]

image-20240117201643939

第五步:

nums1[p1] < nums2[p2]nums1[p--] = nums2[p2--]

image-20240117201658989

此时p2<0,退出。

参考代码

C++
#include <iostream>
#include <vector>class Solution {
public:void merge(std::vector<int>& nums1, int m, std::vector<int>& nums2, int n) {int p1 = m - 1; // 指向 nums1 的末尾int p2 = n - 1; // 指向 nums2 的末尾int p = m + n - 1; // 指向合并后的 nums1 的末尾// 从后向前遍历合并 nums2 到 nums1 中while (p1 >= 0 && p2 >= 0) {if (nums1[p1] >= nums2[p2]) {nums1[p--] = nums1[p1--];} else {nums1[p--] = nums2[p2--];}}// 将 nums2 中剩余的元素(如果有)合并到 nums1 中while (p2 >= 0) {nums1[p--] = nums2[p2--];}}
};int main() {Solution solution;std::vector<int> nums1 = {1, 2, 3, 0, 0, 0};std::vector<int> nums2 = {2, 5, 6};int m = 3;int n = 3;solution.merge(nums1, m, nums2, n);// 输出合并后的 nums1for (int num : nums1) {std::cout << num << " ";}std::cout << std::endl;return 0;
}
Java
import java.util.Arrays;class Solution {public void merge(int[] nums1, int m, int[] nums2, int n) {int p1 = m - 1; // 指向 nums1 的末尾int p2 = n - 1; // 指向 nums2 的末尾int p = m + n - 1; // 指向合并后的 nums1 的末尾// 从后向前遍历合并 nums2 到 nums1 中while (p1 >= 0 && p2 >= 0) {if (nums1[p1] >= nums2[p2]) {nums1[p--] = nums1[p1--];} else {nums1[p--] = nums2[p2--];}}// 将 nums2 中剩余的元素(如果有)合并到 nums1 中while (p2 >= 0) {nums1[p--] = nums2[p2--];}}
}public class Main {public static void main(String[] args) {Solution solution = new Solution();int[] nums1 = {1, 2, 3, 0, 0, 0};int[] nums2 = {2, 5, 6};int m = 3;int n = 3;solution.merge(nums1, m, nums2, n);// 输出合并后的 nums1System.out.println(Arrays.toString(nums1));}
}
Python
class Solution:def merge(self, nums1, m, nums2, n):p1 = m - 1  # 指向 nums1 的末尾p2 = n - 1  # 指向 nums2 的末尾p = m + n - 1  # 指向合并后的 nums1 的末尾# 从后向前遍历合并 nums2 到 nums1 中while p1 >= 0 and p2 >= 0:if nums1[p1] >= nums2[p2]:nums1[p] = nums1[p1]p1 -= 1else:nums1[p] = nums2[p2]p2 -= 1p -= 1# 将 nums2 中剩余的元素(如果有)合并到 nums1 中while p2 >= 0:nums1[p] = nums2[p2]p -= 1p2 -= 1if __name__ == "__main__":solution = Solution()nums1 = [1, 2, 3, 0, 0, 0]nums2 = [2, 5, 6]m = 3n = 3solution.merge(nums1, m, nums2, n)print(nums1)

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

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

相关文章

【讲解下Fiddler的安装和使用】

&#x1f3a5;博主&#xff1a;程序员不想YY啊 &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 ✨希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出…

设计模式学习笔记 - 设计模式与范式 -行为型:11.迭代器模式(下):如何设计实现一个支持“快照”功能的Iterator

概述 前两篇文章&#xff0c;学习了迭代器模式的原理、实现&#xff0c;并分析了在遍历集合的同时增删元素集合&#xff0c;产生不可预期结果的原因及应对策略。 本章&#xff0c;再来看这样一个问题&#xff1a; 如何实现一个支持 “快照” 功能的迭代器&#xff1f; 这个问…

【IT资质合集】CMMI软件能力成熟度介绍,一定不要错过!

近几年&#xff0c;IT企业在市场上逐渐增多&#xff0c;很多企业为了能在市场上取得一定的发展&#xff0c;那办理几项企业相关资质是不可缺少的&#xff0c;任何一项资质对于企业来说都是具有一定优势的&#xff0c;同时也会更加利于企业的发展。 企业常办理的IT资质类 ✅ C…

C语言每日一题(67)长度最小的子数组

题目链接 209 长度最小的子数组 题目描述 给定一个含有 n 个正整数的数组和一个正整数 target 。 找出该数组中满足其总和大于等于 target 的长度最小的 连续子数组 [numsl, numsl1, ..., numsr-1, numsr] &#xff0c;并返回其长度。如果不存在符合条件的子数组&#xff0c…

宠物救助系统|基于Springboot和vue的流浪猫狗救助救援系统设计与实现(源码+数据库+文档)

宠物救助目录 基于Springboot和vue的流浪猫狗救助救援系统设计与实现 一、前言 二、系统设计 三、系统功能设计 1、前台&#xff1a; 2、后台 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取&#xff1a; 博主介绍&#xff1a;✌…

面试经典算法系列之链表2 -- 环形链表

面试经典算法8-环形链表 LeetCode.141 公众号&#xff1a;阿Q技术站 问题描述 给你一个链表的头节点 head &#xff0c;判断链表中是否有环。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&am…

flask 后端 + 微信小程序和网页两种前端:调用硬件(相机和录音)和上传至服务器

选择 flask 作为后端&#xff0c;因为后续还需要深度学习模型&#xff0c;python 语言最适配&#xff1b;而 flask 框架轻、学习成本低&#xff0c;所以选 flask 作为后端框架。 微信小程序封装了调用手机硬件的 api&#xff0c;通过它来调用手机的摄像头、录音机&#xff0c;…

【C++成长记】C++入门 |函数重载、引用、内联函数

&#x1f40c;博主主页&#xff1a;&#x1f40c;​倔强的大蜗牛&#x1f40c;​ &#x1f4da;专栏分类&#xff1a;C❤️感谢大家点赞&#x1f44d;收藏⭐评论✍️ 目录 一、函数重载 1、函数重载概念 二、引用 1、引用概念 2、引用特性 3、常引用 4、使用场景 5、…

数据库之DCL操作(用户、访问权限。)

DCL英文全称是Data control language(数据控制语言)&#xff0c;用来管理数据库用户、控制数据库的访问权限。 1.管理用户 1.1查询用户 select * from mysql.user; 其中 Host代表当前用户访问的主机&#xff0c;如果为localhost&#xff0c;仅代表只能够在当前本机访问&…

Synergy错误: NOTE: Cursor is locked to screen, check Scroll Lock key

错误&#xff1a; NOTE: Cursor is locked to screen, check Scroll Lock key NOTE: Cursor is locked to screen, check Scroll Lock key NOTE: Cursor is locked to screen, check Scroll Lock key NOTE: Cursor is locked to screen, check Scroll Lock key NOTE: Cursor is…

深入浅出 -- 系统架构之微服务中OpenFeign最佳实践

前面我们讲了一下 Ribbon 和 RestTemplate 实现服务端通信的方法&#xff0c;Ribbon 提供了客户端负载均衡&#xff0c;而 RestTemplate 则对 http 进行封装&#xff0c;简化了发送请求的流程&#xff0c;两者互相配合&#xff0c;构建了服务间的高可用通信。 但在使用后也会发…

谁在投资“元素周期表”? 顶级芯片制造商“军备竞赛”

有色和商品基金的大买家何在 投资A股&#xff0c;有时候投资的也是一种“玄妙”的境界。 你需要复习金融知识、复习经济知识&#xff0c;复习科技知识&#xff0c;学习财政学、学习人口学、学习传染病学。 但这些可能还不够。 你能想象么有朝一日&#xff0c;你会回头复习中…

Flask项目如何在测试环境和生产环境部署上线

前言 最近在使用Flask框架&#xff0c;写一个小项目&#xff0c;在项目部署启动后&#xff0c;出现了以下这段提示&#xff0c;这段提示的意思是&#xff0c;该启动方式适用于开发环境中&#xff0c;生产环境要使用WSGI服务器。 WARNING: This is a development server. Do no…

阿里云大学考试Java中级题目及解析-java中级

阿里云大学考试Java中级题目及解析 1.servlet释放资源的方法是&#xff1f; A.int()方法 B.service()方法 C.close() 方法 D.destroy()方法 D servlet释放资源的方法是destroy() 2.order by与 group by的区别&#xff1f; A.order by用于排序&#xff0c;group by用于排序…

从0到1一步一步玩转openEuler--02 openEuler操作系统的安装

从0到1一步一步玩转openEuler–02 openEuler操作系统的安装 安装地址&#xff1a;https://www.jianshu.com/p/f8b8c7b4cc11

OSCP靶场--Zino

OSCP靶场–Zino 考点(CVE-2019-9581 RCE 定时任务脚本可写提权) 1.nmap扫描 ##┌──(root㉿kali)-[~/Desktop] └─# nmap 192.168.173.64 -sV -sC -Pn --min-rate 2500 -p- Starting Nmap 7.92 ( https://nmap.org ) at 2024-04-10 04:18 EDT Nmap scan report for 192.…

自定义注解进行数据转换

前言&#xff1a; Java注解是一种元数据机制&#xff0c;可用于方法&#xff0c;字段&#xff0c;类等程序上以提供关于这些元素的额外信息。 以下内容是我自己写的一个小测试的demo,参考该文章进行编写&#xff1a;https://blog.csdn.net/m0_71621983/article/details/1318164…

【linux】基础IO(四)

在上一篇基础IO中我们主要讲述了文件再磁盘中的存储&#xff0c;当然我们说的也都只是预备知识&#xff0c;为这一篇的文件系统进行铺垫。 目录 搭文件系统的架子&#xff1a;填补细节&#xff1a;inode&#xff1a;datablock[]: 更上层的理解&#xff1a; 搭文件系统的架子&a…

dynamicreports示例

1. 简单段落文本报表 //标题样式StyleBuilder titleStyle DynamicReports.stl.style().setHorizontalTextAlignment(HorizontalTextAlignment.CENTER)//设置对齐方式.setFontSize(50)//设置字体.setBackgroundColor(Color.CYAN);//设置背景颜色//段落样式StyleBuilder paragra…

uniapp 2.0可视化工具:创建与管理Vue文件的实践之旅

引言 在前端开发领域中&#xff0c;Vue以其简洁、易上手的特点&#xff0c;受到了广大开发者的青睐。随着uniapp的不断发展&#xff0c;越来越多的开发者开始利用uniapp的可视化工具来创建和管理Vue文件&#xff0c;以提高开发效率。本文将详细介绍如何使用uniapp 2.0可视化工…