力扣面试经典算法150题:合并两个有序数组

算法

本篇开始,正式进入算法刷题篇。

题目来源于力扣面试经典150题。

题目链接:https://leetcode.cn/studyplan/top-interview-150/

合并两个有序数组

题目选自150题中的数组/字符串一类,题目难度:简单。

题目描述

给定两个按非递减顺序排列的整数数组 nums1nums2,以及两个整数 mn,分别代表 nums1nums2 中的实际元素数量。将 nums2 中的所有元素合并到 nums1 中,使得 nums1 成为一个有序数组。

  • 注意:
    • 数组 nums1 的长度等于 m + n,意味着它有足够的空间容纳 nums2 中的所有元素。
    • 不要使用额外的数组空间,你应该在原地修改输入数组并在使用 O(1) 额外空间的条件下完成此操作。
  • 示例:
    • 输入:
      nums1 = [1,2,3,0,0,0], m = 3
      nums2 = [2,5,6], n = 3
    • 输出:
      nums1 = [1,2,2,3,5,6]
  • 解释:
    • 数组 nums1 的前 m 个元素 [1,2,3] 是已排序的。
    • 数组 nums2 的所有元素 [2,5,6] 也是已排序的。
    • 合并后,nums1 应该包含 [1,2,2,3,5,6],并且仍然保持有序状态。

题目分析

题目要求我们在不使用额外空间的情况下,将两个已排序的数组合并成一个新的有序数组。

由于 nums1 已经预留了足够的空间来存放 nums2 中的元素,因此我们可以通过某种方式直接在 nums1 上进行操作。

为了保证合并后的数组仍然有序,我们需要找到一种方法,能够确保较大的元素被放置在数组的后面,较小的元素被放置在前面。对于这种数组的排序算法,如果是递增排序(从小到大),我们采取从数组尾部开始比较元素并放置;反之如果是递减排序(从大到小),我们就采取从数组头部开始进行元素的比较与放置。

解题思路

  1. 递增排序的数组,我们从尾部开始比较与放置元素,所以我们需要定义出三个变量,分别作为nums1nums2nums1(合并后)数组的最大索引值。
  2. 很明显算法必须必须使用循环语句实现,那么循环条件必须是能同时拿到两个初始数组的元素才能进行循环,因为两个数组的元素需要作比较。
  3. 算法的核心内容也就是循环语句中的内容是对两个数组的元素大小比较,将大的值放入到合成后的数组中,就是nums1
  4. 由于两个数组长度不等,可能存在一个数组元素取完,另一个元素未取完的场景,因此我们需要对长度较长的数组单独做一次循环,直接较长的数组的元素也取完。在本题中,由于两数组合并到nums1数组,所以nums1的长度是大于nums2的长度。注意这里指的是长度,而不是实际的元素个数。

实际算法代码

根据以上分析和思路,我们可以写出以下代码:

java 复制代码
    public static void main(String[] args) {
        // 示例数据
        int[] nums1 = {1, 2, 3, 0, 0, 0};
        int m = 3;
        int[] nums2 = {2, 5, 6};
        int n = 3;

        // 调用合并方法
        merge(nums1, m, nums2, n);

        // 输出合并后的数组
        System.out.println(Arrays.toString(nums1));
    }

	 /**
     * 合并两个有序数组
     *
     * @param nums1 第一个有序数组
     * @param m     nums1 中实际元素的数量
     * @param nums2 第二个有序数组
     * @param n     nums2 中实际元素的数量
     */
    private static void merge(int[] nums1, int m, int[] nums2, int n) {
    	// nums1 的最后一个实际元素的索引
        int i = m - 1; 
        // nums2 的最后一个元素的索引
        int j = n - 1; 
        // 合并后数组的最后一个位置的索引
        int k = m + n - 1; 

        // 从后向前比较和放置元素
        while (i >= 0 && j >= 0) {
        	// 取较大的元素放入合并后的数组中,并且注意k的值要减1,因为我们是从尾部开始处理,要往上一个位置放元素
        	// 被取出元素的数组索引要减1,下次取的时候就是上一个位置元素
            if (nums1[i] > nums2[j]) {
                nums1[k] = nums1[i];
                k--;
                i--;
            } else {
                nums1[k] = nums2[j];
                k--;
                j--;
            }
        }

        // 如果 nums2 还有剩余元素,则将其复制到 nums1 的前面
        // 解题思路4
        while (j >= 0) {
            nums1[k] = nums2[j];
            k--;
            j--;
        }
    }

验证

算法编码编写完毕,进行验证。启动main函数,效果如下:

力扣执行:

完美通过!

相关推荐
尘浮生3 分钟前
Java项目实战II基于微信小程序的校运会管理系统(开发文档+数据库+源码)
java·开发语言·数据库·微信小程序·小程序·maven·intellij-idea
小白不太白9507 分钟前
设计模式之 模板方法模式
java·设计模式·模板方法模式
Tech Synapse9 分钟前
Java根据前端返回的字段名进行查询数据的方法
java·开发语言·后端
xoxo-Rachel16 分钟前
(超级详细!!!)解决“com.mysql.jdbc.Driver is deprecated”警告:详解与优化
java·数据库·mysql
乌啼霜满天24918 分钟前
JDBC编程---Java
java·开发语言·sql
肥猪猪爸28 分钟前
使用卡尔曼滤波器估计pybullet中的机器人位置
数据结构·人工智能·python·算法·机器人·卡尔曼滤波·pybullet
色空大师30 分钟前
23种设计模式
java·开发语言·设计模式
闲人一枚(学习中)31 分钟前
设计模式-创建型-建造者模式
java·设计模式·建造者模式
readmancynn40 分钟前
二分基本实现
数据结构·算法