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

这篇具有很好参考价值的文章主要介绍了【经典算法】 leetcode88.合并排序的数组(Java/C/Python3实现含注释说明,Easy)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

  • 作者主页: 🔗进朱者赤的博客

  • 精选专栏:🔗经典算法

  • 作者简介:阿里非典型程序员一枚 ,记录在大厂的打怪升级之路。 一起学习Java、大数据、数据结构算法(公众号同名

  • ❤️觉得文章还不错的话欢迎大家点赞👍➕收藏⭐️➕评论,💬支持博主,记得点个大大的关注,持续更新🤞
    ————————————————-

  • 标签(题目类型):数组,双指针

题目描述

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

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

欢迎一键三连(关注+点赞+收藏),技术的路上一起加油!!!代码改变世界

  • 关于我:阿里非典型程序员一枚 ,记录在大厂的打怪升级之路。 一起学习Java、大数据、数据结构算法(公众号同名),回复暗号,更能获取学习秘籍和书籍等

  • —⬇️欢迎关注下面的公众号:进朱者赤,认识不一样的技术人。⬇️—文章来源地址https://www.toymoban.com/news/detail-855287.html

到了这里,关于【经典算法】 leetcode88.合并排序的数组(Java/C/Python3实现含注释说明,Easy)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Leetcode. 88合并两个有序数组

    合并两个有序数组 核心思路: 依次比较,取较小值放入新数组中 i 遍历nums1 , j 遍历nums2 ,取较小值放入nums3中 那如果nums[i] 和nums[j]中相等,随便放一个到nums3 那如果nums[i] 和nums[j]中相等,随便放一个到nums3 此时 nums1 中的元素已经走完了,那么直接把 nums2 中剩下的元素拿到

    2023年04月08日
    浏览(99)
  • LeetCode88——合并两个有序数组

    给你两个按 非递减顺序 排列的整数数组 nums1 和 nums2,另有两个整数 m 和 n ,分别表示 nums1 和 nums2 中的元素数目。 请你 合并 nums2 到 nums1 中,使合并后的数组同样按 非递减顺序 排列。 注意:最终,合并后数组不应由函数返回,而是存储在数组 nums1 中。 为了应对这种情况

    2024年02月08日
    浏览(41)
  • 【Leetcode】88.合并两个有序数组

    给你两个按 非递减顺序 排列的整数数组 nums1 和 nums2 ,另有两个整数 m 和 n ,分别表示 nums1 和 nums2 中的元素数目。 请你 合并 nums2 到 nums1 中,使合并后的数组同样按 非递减顺序 排列。 注意 :最终,合并后数组不应由函数返回,而是存储在数组 nums1 中。为了应对这种情况

    2024年02月12日
    浏览(43)
  • LeetCode_88. 合并两个有序数组

    目录 题目描述 思路分析 我的题解 给你两个按 非递减顺序 排列的整数数组 nums1 和 nums2,另有两个整数 m 和 n ,分别表示 nums1 和 nums2 中的元素数目。 请你 合并 nums2 到 nums1 中,使合并后的数组同样按 非递减顺序 排列。 注意:最终,合并后数组不应由函数返回,而是存储

    2023年04月15日
    浏览(48)
  • ​LeetCode解法汇总88. 合并两个有序数组

    https://github.com/September26/java-algorithms 给你两个按  非递减顺序  排列的整数数组  nums1   和  nums2 ,另有两个整数  m  和  n  ,分别表示  nums1  和  nums2  中的元素数目。 请你  合并   nums2   到  nums1  中,使合并后的数组同样按  非递减顺序  排列。 注意: 最终,合并

    2024年02月12日
    浏览(45)
  • 88. 合并两个有序数组、Leetcode的Python实现

    博客主页:🏆李歘歘的博客 🏆 🌺每天不定期分享一些包括但不限于计算机基础、算法、后端开发相关的知识点,以及职场小菜鸡的生活。🌺 💗点关注不迷路,总有一些📖知识点📖是你想要的💗 ⛽️今天的内容是     Leetcode  88. 合并两个有序数组        ⛽️💻💻💻

    2024年02月06日
    浏览(55)
  • 力扣每日一道系列 --- LeetCode 88. 合并两个有序数组

    📷 江池俊: 个人主页 🔥个人专栏: ✅数据结构探索 ✅LeetCode每日一道 🌅 有航道的人,再渺小也不会迷途。 LeetCode 88. 合并两个有序数组 首先创建一个临时数组,其大小为第一个数组的大小(即nums1Size),其作用主要是。 通过循环遍历两个数组,将两个数组元素比较后较

    2024年02月04日
    浏览(51)
  • 【经典算法】LeetCode 5: 最长回文子串(Java/C/Python3实现含注释说明,Medium)

    标签(题目类型):回文串、动态规划 原题:LeetCode 5 思路 Dynamic Programming(DP) 动态规划是一种将问题分解成子问题并分别计算的优化技术。对于回文子串,我们可以使用动态规划来解决。 对于一个子串而言,如果它是回文串,并且长度大于 2,那么将它首尾的两个字母去除之后

    2024年04月14日
    浏览(66)
  • LeetCode150道面试经典题-合并两个有序数组(简单)

    题目: 给你两个按 非递减顺序 排列的整数数组  nums1 和 nums2 ,另有两个整数 m 和 n ,分别表示 nums1 和 nums2 中的元素数目。 请你 合并 nums2 到 nums1 中,使合并后的数组同样按 非递减顺序 排列。 注意: 最终,合并后数组不应由函数返回,而是存储在数组 nums1 中。为了应对

    2024年02月14日
    浏览(44)
  • 【经典LeetCode算法题目专栏分类】【第6期】二分查找系列:x的平方根、有效完全平方数、搜索二位矩阵、寻找旋转排序数组最小值

    《博主简介》 小伙伴们好,我是阿旭。专注于人工智能AI、python、计算机视觉相关分享研究。 ✌ 更多学习资源,可关注公-仲-hao:【阿旭算法与机器学习】,共同学习交流~ 👍 感谢小伙伴 们点赞、关注! class   Solution :      def   mySqrt ( self ,  x :   int )   -   int :       

    2024年02月04日
    浏览(65)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包