【面试经典 150 | 二分查找】寻找两个正序数组的中位数

2024-04-18 08:12

本文主要是介绍【面试经典 150 | 二分查找】寻找两个正序数组的中位数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 写在前面
  • Tag
  • 题目来源
  • 题目解读
    • 方法一:朴素
    • 方法二:二分查找【寻找第k小元素】
  • 写在最后

写在前面

本专栏专注于分析与讲解【面试经典150】算法,两到三天更新一篇文章,欢迎催更……

专栏内容以分析题目为主,并附带一些对于本题涉及到的数据结构等内容进行回顾与总结,文章结构大致如下,部分内容会有增删:

  • Tag:介绍本题牵涉到的知识点、数据结构;
  • 题目来源:贴上题目的链接,方便大家查找题目并完成练习;
  • 题目解读:复述题目(确保自己真的理解题目意思),并强调一些题目重点信息;
  • 解题思路:介绍一些解题思路,每种解题思路包括思路讲解、实现代码以及复杂度分析;
  • 知识回忆:针对今天介绍的题目中的重点内容、数据结构进行回顾总结。

Tag

【数组】【二分查找】【双指针】


题目来源

4. 寻找两个正序数组的中位数


题目解读

方法一:朴素

给出两个有序数组,要求找出两个有序数组合并后数组的中位数。朴素的方法是直接拼接两个数组,然后排序,取出 “中间” 位置的元素,即为中位数。时间和空间复杂度分别为 O ( ( m + n ) l o g ( m + n ) ) O((m+n)log(m+n)) O((m+n)log(m+n)) O ( l o g ( m + n ) O(log(m+n) O(log(m+n)

如果利用 88. 合并两个有序数组 中的双指针解法合并数组,再找出 “中间” 位置,分别可以把时空复杂度降到 O ( m + n ) O(m+n) O(m+n) O ( m + n ) O(m+n) O(m+n)

如果不执着于合并两个有序数组,可以利用双指针直接找到中位数的位置。由于两个数组的长度已知,因此中位数对应的两个数组的下标之和也是已知的。维护两个指针,初始时分别指向两个数组的下标 000 的位置,每次将指向较小值的指针后移一位(如果一个指针已经到达数组末尾,则只需要移动另一个数组的指针),直到到达中位数的位置。(双指针直接找中位数参考自 官方题解)

这样的空间复杂度可以降到 O ( 1 ) O(1) O(1),但是时间复杂度依然是 O ( m + n ) O(m+n) O(m+n)

题目中要求对数时间的解法,通常就是二分查找了。

方法二:二分查找【寻找第k小元素】

寻找第 k 小元素

根据中位数的定义,当 m+n 是奇数时,中位数是两个有序数组中的第 (m+n)/2 个元素(计数从1开始),当 m+n 是偶数时,中位数是两个有序数组中的第 (m+n)/2 个元素和第 (m+n)/2+1 个元素的平均值。因此,这道题可以转化成寻找两个有序数组中的第 k 小的数,其中 k 为 (m+n)/2 或 (m+n)/2+1。

如何二分寻找第 k 小元素

假设两个数组分别为 A 和 B。要寻找第 k 小元素,我们可以比较 A[k / 2 - 1]B[k / 2 - 1]:

  • 如果 A[k / 2 - 1] < B[k / 2 - 1],则比 A[k / 2 - 1] 小的数最多只有 A 的前 k/ 2 - 1 个数 和 B 的前 k/ 2 - 1 个数,即最多共计 k - 2 个数,那么 A[0]A[k/2 - 1] 不可能是第 k 个数,可以全部排除。
  • 同理, A[k / 2 - 1] > B[k / 2 - 1] 时,可以排除 B[0]B[k/2 - 1]
  • A[k / 2 - 1] = B[k / 2 - 1] 时,可以归入以上任意一种情况。

三种情况需要特殊处理:

  • 如果 A[k/2−1] 或者 B[k/2−1] 越界,那么我们可以选取对应数组中的最后一个元素。在这种情况下,我们必须根据排除数的个数减少 k 的值,而不能直接将 k 减去 k/2
  • 如果一个数组为空,说明该数组中的所有元素都被排除,我们可以直接返回另一个数组中第 k 小的元素。
  • 如果 k=1,我们只要返回两个数组首元素的最小值即可。

代码

class Solution {
public:int getKthElement(const vector<int>& nums1, const vector<int>& nums2, int k) {int m = nums1.size(), n = nums2.size();int idx1 = 0, idx2 = 0; // 表示排除后新的开始位置while (true) {if (idx1 == m) {    // nums1 都被排除完return nums2[idx2 + k - 1];}if (idx2 == n) {    // nums2 都被排除完return nums1[idx1 + k - 1];}if (k == 1) {       // 找出第 1 小的元素return min(nums1[idx1], nums2[idx2]);}int newIdx1 = min(idx1 + k/2 - 1, m-1);int newIdx2 = min(idx2 + k/2 - 1, n-1);if (nums1[newIdx1] <= nums2[newIdx2]) {k -= newIdx1 - idx1 + 1;idx1 = newIdx1 + 1;}else {k -= newIdx2 - idx2 + 1;idx2 = newIdx2 + 1;}}}double findMedianSortedArrays(vector<int>& nums1, vector<int>& nums2) {int totalLen = nums1.size() + nums2.size();if (totalLen & 1) {   // 奇数return getKthElement(nums1, nums2, (totalLen + 1) / 2);}else {return (getKthElement(nums1, nums2, (totalLen + 1) / 2) + getKthElement(nums1, nums2, (totalLen + 1) / 2 + 1)) / 2.0; }}
};

复杂度分析

时间复杂度: O ( l o g ( m + n ) ) O(log(m+n)) O(log(m+n)) m m m n n n 分别为数组 nums1nums2 的长度。每一轮循环都会将查找缩小一半。

空间复杂度: O ( 1 ) O(1) O(1)


写在最后

如果您发现文章有任何错误或者对文章有任何疑问,欢迎私信博主或者在评论区指出 💬💬💬。

如果大家有更优的时间、空间复杂度的方法,欢迎评论区交流。

最后,感谢您的阅读,如果有所收获的话可以给我点一个 👍 哦。

这篇关于【面试经典 150 | 二分查找】寻找两个正序数组的中位数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/914233

相关文章

MySQL JSON 查询中的对象与数组技巧及查询示例

《MySQLJSON查询中的对象与数组技巧及查询示例》MySQL中JSON对象和JSON数组查询的详细介绍及带有WHERE条件的查询示例,本文给大家介绍的非常详细,mysqljson查询示例相关知... 目录jsON 对象查询1. JSON_CONTAINS2. JSON_EXTRACT3. JSON_TA

C++作用域和标识符查找规则详解

《C++作用域和标识符查找规则详解》在C++中,作用域(Scope)和标识符查找(IdentifierLookup)是理解代码行为的重要概念,本文将详细介绍这些规则,并通过实例来说明它们的工作原理,需... 目录作用域标识符查找规则1. 普通查找(Ordinary Lookup)2. 限定查找(Qualif

JAVA数组中五种常见排序方法整理汇总

《JAVA数组中五种常见排序方法整理汇总》本文给大家分享五种常用的Java数组排序方法整理,每种方法结合示例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录前言:法一:Arrays.sort()法二:冒泡排序法三:选择排序法四:反转排序法五:直接插入排序前言:几种常用的Java数组排序

C#实现查找并删除PDF中的空白页面

《C#实现查找并删除PDF中的空白页面》PDF文件中的空白页并不少见,因为它们有可能是作者有意留下的,也有可能是在处理文档时不小心添加的,下面我们来看看如何使用Spire.PDFfor.NET通过C#... 目录安装 Spire.PDF for .NETC# 查找并删除 PDF 文档中的空白页C# 添加与删

Git可视化管理工具(SourceTree)使用操作大全经典

《Git可视化管理工具(SourceTree)使用操作大全经典》本文详细介绍了SourceTree作为Git可视化管理工具的常用操作,包括连接远程仓库、添加SSH密钥、克隆仓库、设置默认项目目录、代码... 目录前言:连接Gitee or github,获取代码:在SourceTree中添加SSH密钥:Cl

pandas中位数填充空值的实现示例

《pandas中位数填充空值的实现示例》中位数填充是一种简单而有效的方法,用于填充数据集中缺失的值,本文就来介绍一下pandas中位数填充空值的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录什么是中位数填充?为什么选择中位数填充?示例数据结果分析完整代码总结在数据分析和机器学习过程中,处理缺失数

Java数组初始化的五种方式

《Java数组初始化的五种方式》数组是Java中最基础且常用的数据结构之一,其初始化方式多样且各具特点,本文详细讲解Java数组初始化的五种方式,分析其适用场景、优劣势对比及注意事项,帮助避免常见陷阱... 目录1. 静态初始化:简洁但固定代码示例核心特点适用场景注意事项2. 动态初始化:灵活但需手动管理代

数据库面试必备之MySQL中的乐观锁与悲观锁

《数据库面试必备之MySQL中的乐观锁与悲观锁》:本文主要介绍数据库面试必备之MySQL中乐观锁与悲观锁的相关资料,乐观锁适用于读多写少的场景,通过版本号检查避免冲突,而悲观锁适用于写多读少且对数... 目录一、引言二、乐观锁(一)原理(二)应用场景(三)示例代码三、悲观锁(一)原理(二)应用场景(三)示例

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的