【经典算法】 leetcode88.合并排序的数组(Java/C/Python3实现含注释说明,Easy)

简介: 【经典算法】 leetcode88.合并排序的数组(Java/C/Python3实现含注释说明,Easy)

题目描述

给你两个按 非递减顺序 排列的整数数组 nums1 和 nums2,另有两个整数 m 和 n ,分别表示 nums1 和 nums2 中的元素数目。
请你 合并 nums2 到 nums1 中,使合并后的数组同样按 非递减顺序 排列。
注意:最终,合并后数组不应由函数返回,而是存储在数组 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
 
进阶:你可以设计实现一个时间复杂度为 O(m + n) 的算法解决此问题吗?

原题:LeetCode 88

思路及实现

方式一:从后向前双指针

思路

我们可以使用双指针,分别从两个数组的末尾开始,将较大的数依次放入 nums1 的末尾。因为 nums1 的空间足够大,所以可以直接将 nums2 中的元素放入 nums1 中。当 nums2 的指针到达末尾时,nums1 中剩下的元素就无需移动了,因为它们已经是排好序的。

代码实现

Java版本
public void merge(int[] nums1, int m, int[] nums2, int n) {
    int i = m - 1; // nums1 有效元素末尾指针
    int j = n - 1; // nums2 末尾指针
    int k = m + n - 1; // 合并后数组末尾指针
    
    while (j >= 0) {
        if (i >= 0 && nums1[i] > nums2[j]) {
            nums1[k--] = nums1[i--];
        } else {
            nums1[k--] = nums2[j--];
        }
    }
}

说明:

  • 初始化三个指针,i 指向 nums1 的有效元素末尾,j 指向 nums2 的末尾,k 指向合并后数组的末尾。
  • 当 j >= 0 时,说明 nums2 中还有元素未处理,继续循环。
  • 比较 nums1[i] 和 nums2[j] 的大小,将较大的数放入 nums1[k],并将对应指针前移。
  • 循环结束后,nums1 中的元素已经是合并后的排序数组。
C语言版本
void merge(int* nums1, int nums1Size, int m, int* nums2, int nums2Size, int n){
    int i = m - 1;
    int j = n - 1;
    int k = m + n - 1;
    
    while (j >= 0) {
        if (i >= 0 && nums1[i] > nums2[j]) {
            nums1[k--] = nums1[i--];
        } else {
            nums1[k--] = nums2[j--];
        }
    }
}

说明:

  • C语言版本的实现与 Java 版本类似,只是参数传递和数组访问的方式略有不同。
Python3版本
def merge(nums1: List[int], m: int, nums2: List[int], n: int) -> None:
    i = m - 1
    j = n - 1
    k = m + n - 1
    
    while j >= 0:
        if i >= 0 and nums1[i] > nums2[j]:
            nums1[k] = nums1[i]
            i -= 1
        else:
            nums1[k] = nums2[j]
            j -= 1
        k -= 1

说明:

  • Python 版本的实现逻辑与 Java 和 C 版本相同,只是语法上有些差异。

复杂度分析

  • 时间复杂度:O(m + n),其中 m 和 n 分别是 nums1 和 nums2 的长度。需要遍历两个数组中的所有元素。
  • 空间复杂度:O(1),除了原数组 nums1 外,没有使用额外的空间。

方式二:从前往后双指针(需要额外空间)

思路

我们创建一个新的数组 temp,其长度为 m + n(即 nums1nums2 合并后的长度)。然后,我们使用两个指针分别遍历 nums1nums2,将较小的元素依次放入 temp 数组中,直到其中一个数组遍历完。最后,将另一个数组中剩余的元素(如果有的话)直接复制到 temp 数组的末尾。最后,将 temp 数组的内容复制回 nums1

代码实现

Java版本
public void merge(int[] nums1, int m, int[] nums2, int n) {
    int[] temp = new int[m + n];
    int i = 0, j = 0, k = 0;
    
    while (i < m && j < n) {
        if (nums1[i] <= nums2[j]) {
            temp[k++] = nums1[i++];
        } else {
            temp[k++] = nums2[j++];
        }
    }
    
    while (i < m) {
        temp[k++] = nums1[i++];
    }
    
    while (j < n) {
        temp[k++] = nums2[j++];
    }
    
    // 将 temp 数组的内容复制回 nums1
    for (i = 0; i < m + n; i++) {
        nums1[i] = temp[i];
    }
}

说明:

  • 初始化三个指针 ijk,分别用于遍历 nums1nums2 和填充 temp 数组。
  • 使用 while 循环比较 nums1[i]nums2[j] 的大小,将较小的数放入 temp[k] 中,并移动对应的指针。
  • 当其中一个数组遍历完后,将另一个数组中剩余的元素直接复制到 temp 数组的末尾。
  • 最后,将 temp 数组的内容复制回 nums1
C语言版本
void merge(int* nums1, int nums1Size, int m, int* nums2, int nums2Size, int n) {
    int* temp = (int*)malloc((m + n) * sizeof(int));
    int i = 0, j = 0, k = 0;
    
    while (i < m && j < n) {
        if (nums1[i] <= nums2[j]) {
            temp[k++] = nums1[i++];
        } else {
            temp[k++] = nums2[j++];
        }
    }
    
    while (i < m) {
        temp[k++] = nums1[i++];
    }
    
    while (j < n) {
        temp[k++] = nums2[j++];
    }
    
    // 将 temp 数组的内容复制回 nums1
    for (i = 0; i < m + n; i++) {
        nums1[i] = temp[i];
    }
    
    free(temp); // 释放临时数组的空间
}

说明:

  • C语言版本与 Java 版本逻辑相似,但需要注意内存分配和释放。使用 malloc 分配临时数组 temp 的空间,并在使用完毕后使用 free 释放空间。
Python3版本
from typing import List
def merge(nums1: List[int], m: int, nums2: List[int], n: int) -> None:
    temp = [0] * (m + n)
    i, j, k = 0, 0, 0
    
    while i < m and j < n:
        if nums1[i] <= nums2[j]:
            temp[k] = nums1[i]
            i += 1
        else:
            temp[k] = nums2[j]
            j += 1
        k += 1
    
    while i < m:
        temp[k] = nums1[i]
        i += 1
        k += 1
    
    while j < n:
        temp[k] = nums2[j]
        j += 1
        k += 1
    
    # 将 temp 数组的内容复制回nums1
    for i in range(m + n):
    nums1[i] = temp[i]

说明:

  • Python版本使用了列表(List)作为数组,其动态分配内存的特性使得我们无需手动管理内存。
  • 逻辑与其他版本类似,都是将较小的元素依次放入临时数组 temp 中,然后将 temp 的内容复制回 nums1

复杂度分析

  • 时间复杂度:O(m + n),其中 m 和 n 分别是 nums1nums2 的长度。我们需要遍历两个数组的所有元素,并将它们放入临时数组 temp 中,最后再复制回 nums1
  • 空间复杂度:O(m + n),我们需要一个额外的数组 temp 来存储合并后的结果。这个数组的长度是 m + n。

虽然这种方法使用了额外的空间,但它简化了问题,因为我们可以专注于合并两个数组,而无需考虑如何在原地修改 nums1 来容纳 nums2 的元素。在实际应用中,根据具体的性能需求和内存限制,可以选择使用原地修改或者额外空间的方法。

总结

解题方式 优点 缺点 时间复杂度 空间复杂度 其他注意事项
从前往后双指针(原地修改) 无需额外空间,原地修改 逻辑相对复杂,需要正确处理合并细节 O(m + n) O(1) 需要确保 nums1 有足够空间
从前往后双指针(需要额外空间) 逻辑简单,易于理解 需要额外空间存放合并结果 O(m + n) O(m + n) 适用于空间限制不严格的情况

这两种方式都是解决合并两个有序数组问题的有效方法。选择哪种方式取决于具体的性能要求和内存限制。如果内存空间有限,或者希望避免额外的空间开销,可以选择原地修改的方式。如果内存限制不严格,或者希望简化代码逻辑,可以选择使用额外空间的方式。在编写代码时,还需要注意处理边界情况,确保代码的正确性和健壮性。

相似题目

相似题目 难度 链接
两个数组的交集 简单 leetcode-349
两个数组的交集 II 简单 leetcode-350
合并两个有序数组 简单 leetcode-88
合并两个有序链表 中等 leetcode-21
合并K个排序链表 困难 leetcode-23
有序链表转换有序数组 中等 leetcode-148
有序数组的三数之和 中等 leetcode-33
在排序数组中查找元素的第一个和最后一个位置 中等 leetcode-34

这些题目涉及了数组和链表的合并、排序和查找操作,其中一些题目与题目“合并两个有序数组”有类似的解题思路或要求。通过练习这些题目,可以加深对数组和链表操作的理解,提高算法设计和实现的能力。

相关文章
|
3天前
|
存储 Java 索引
Java快速入门之数组、方法
### Java快速入门之数组与方法简介 #### 一、数组 数组是一种容器,用于存储同种数据类型的多个值。定义数组时需指定数据类型,如`int[]`只能存储整数。数组的初始化分为静态和动态两种: - **静态初始化**:直接指定元素,系统自动计算长度,如`int[] arr = {1, 2, 3};` - **动态初始化**:手动指定长度,系统给定默认值,如`int[] arr = new int[3];` 数组访问通过索引完成,索引从0开始,最大索引为`数组.length - 1`。遍历数组常用`for`循环。常见操作包括求和、找最值、统计特定条件元素等。
|
3天前
|
存储 Java 索引
Java基础(六):数组
Java基础(六):数组
Java基础(六):数组
|
1天前
|
存储 Java C++
Java数组:静态初始化与动态初始化详解
本文介绍了Java中数组的定义、特点及初始化方式。
26 12
|
3月前
|
存储 缓存 算法
提高 Java 数组性能的方法
【10月更文挑战第19天】深入探讨了提高 Java 数组性能的多种方法。通过合理运用这些策略,我们可以在处理数组时获得更好的性能表现,提升程序的运行效率。
52 2
|
4天前
|
算法 数据安全/隐私保护 计算机视觉
基于Retinex算法的图像去雾matlab仿真
本项目展示了基于Retinex算法的图像去雾技术。完整程序运行效果无水印,使用Matlab2022a开发。核心代码包含详细中文注释和操作步骤视频。Retinex理论由Edwin Land提出,旨在分离图像的光照和反射分量,增强图像对比度、颜色和细节,尤其在雾天条件下表现优异,有效解决图像去雾问题。
|
4天前
|
算法 数据可视化 安全
基于DWA优化算法的机器人路径规划matlab仿真
本项目基于DWA优化算法实现机器人路径规划的MATLAB仿真,适用于动态环境下的自主导航。使用MATLAB2022A版本运行,展示路径规划和预测结果。核心代码通过散点图和轨迹图可视化路径点及预测路径。DWA算法通过定义速度空间、采样候选动作并评估其优劣(目标方向性、障碍物距离、速度一致性),实时调整机器人运动参数,确保安全避障并接近目标。
|
14天前
|
算法 数据安全/隐私保护
室内障碍物射线追踪算法matlab模拟仿真
### 简介 本项目展示了室内障碍物射线追踪算法在无线通信中的应用。通过Matlab 2022a实现,包含完整程序运行效果(无水印),支持增加发射点和室内墙壁设置。核心代码配有详细中文注释及操作视频。该算法基于几何光学原理,模拟信号在复杂室内环境中的传播路径与强度,涵盖场景建模、射线发射、传播及接收点场强计算等步骤,为无线网络规划提供重要依据。
|
27天前
|
机器学习/深度学习 算法
基于改进遗传优化的BP神经网络金融序列预测算法matlab仿真
本项目基于改进遗传优化的BP神经网络进行金融序列预测,使用MATLAB2022A实现。通过对比BP神经网络、遗传优化BP神经网络及改进遗传优化BP神经网络,展示了三者的误差和预测曲线差异。核心程序结合遗传算法(GA)与BP神经网络,利用GA优化BP网络的初始权重和阈值,提高预测精度。GA通过选择、交叉、变异操作迭代优化,防止局部收敛,增强模型对金融市场复杂性和不确定性的适应能力。
166 80
|
15天前
|
机器学习/深度学习 数据采集 算法
基于GA遗传优化的CNN-GRU-SAM网络时间序列回归预测算法matlab仿真
本项目基于MATLAB2022a实现时间序列预测,采用CNN-GRU-SAM网络结构。卷积层提取局部特征,GRU层处理长期依赖,自注意力机制捕捉全局特征。完整代码含中文注释和操作视频,运行效果无水印展示。算法通过数据归一化、种群初始化、适应度计算、个体更新等步骤优化网络参数,最终输出预测结果。适用于金融市场、气象预报等领域。
基于GA遗传优化的CNN-GRU-SAM网络时间序列回归预测算法matlab仿真
|
15天前
|
算法
基于龙格库塔算法的锅炉单相受热管建模与matlab数值仿真
本设计基于龙格库塔算法对锅炉单相受热管进行建模与MATLAB数值仿真,简化为喷水减温器和末级过热器组合,考虑均匀传热及静态烟气处理。使用MATLAB2022A版本运行,展示自编与内置四阶龙格库塔法的精度对比及误差分析。模型涉及热传递和流体动力学原理,适用于优化锅炉效率。