leetcode热题100.寻找两个正序数组中的中位数

2024-03-27 17:12

本文主要是介绍leetcode热题100.寻找两个正序数组中的中位数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

文章目录

  • 题目
  • 思路
  • 复杂度
  • Code

题目

给定两个大小分别为 m 和 n 的正序(从小到大)数组 nums1 和 nums2。请你找出并返回这两个正序数组的 中位数 。

算法的时间复杂度应该为 O(log (m+n)) 。

示例 1:

输入:nums1 = [1,3], nums2 = [2] 输出:2.00000 解释:合并数组 = [1,2,3] ,中位数 2

示例 2:

输入:nums1 = [1,2], nums2 = [3,4] 输出:2.50000 解释:合并数组 = [1,2,3,4] ,中位数 (2

    1. / 2 = 2.5

思路

定义numus1为较小数组,nums2为较大数组
假设有一条分界线在两个数组nums1和nums2中
这个分界线满足两个条件:

image.png

性质1:
image.png
根据这一性质,我们定义分界线左边的元素个数,可以只二分小的那个nums就可以确定分割线的位置

性质2:
image.png
根据这一性质,我们可以写出二分搜索的判断条件,即左上>右下或者左下>右上都不符合我们的分界线要求

我们发现数组下标i就是nums1分割线左边的元素个数,那么同理 total_left-i 就是nums2分割线右边的元素的位置

我们同时要考虑边界情况,如果 i=0 或者 i=m 或者 j=0 或者 j = n的情况,我们将他们取一个极大的数或者极小的数,为什么这样取,读者可以看最终返回的答案的规则

关于最后的答案,如果 m+n 是奇数,我们只需要取nums1和nums2的分界线左边最大点即可;如果 m + n 是偶数,我们需要取得分界线左边的最大值和分界线右边的最小值的和的二分之一。

复杂度

时间复杂度:

O ( l o g ( m i n ( m , n ) ) O(log(min(m,n)) O(log(min(m,n))

空间复杂度:

O ( 1 ) O(1) O(1)

Code

class Solution:def findMedianSortedArrays(self, nums1: List[int], nums2: List[int]) -> float:# nums1 是较小的数组,nums2是较大的数组if len(nums1) > len(nums2):nums1,nums2 = nums2,nums1m = len(nums1)n = len(nums2)total_left = (m+n+1) // 2# 从小的数组nums1开始二分left = 0rigth = mwhile left<rigth:i = (left+rigth+1) // 2 # nums1的分界线的右边的节点j = total_left - i  # nums2的分界线的右边的节点if nums1[i-1] > nums2[j]: # nums1分界线左边的点小于nums2分界线右边的点rigth = i-1else:left = ii = leftj = total_left - i# 处理边界情况nums1_left = nums1[i-1] if i>0 else float('-inf')nums1_right = nums1[i] if i<m else float('inf') nums2_left = nums2[j-1] if j>0 else float('-inf')nums2_right = nums2[j] if j<n else float('inf')if (m+n) % 2 == 1: # 如果 m+n 是奇数,我们只需要取nums1和nums2的分界线左边最大点即可return max(nums1_left,nums2_left)else: # 如果 m + n 是偶数,我们需要取得分界线左边的最大值和分界线右边的最小值的和的二分之一return (max(nums1_left,nums2_left)+min(nums1_right,nums2_right)) / 2

这篇关于leetcode热题100.寻找两个正序数组中的中位数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

hdu2241(二分+合并数组)

题意:判断是否存在a+b+c = x,a,b,c分别属于集合A,B,C 如果用暴力会超时,所以这里用到了数组合并,将b,c数组合并成d,d数组存的是b,c数组元素的和,然后对d数组进行二分就可以了 代码如下(附注释): #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<que

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

leetcode-24Swap Nodes in Pairs

带头结点。 /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode(int x) { val = x; }* }*/public class Solution {public ListNode swapPairs(L

leetcode-23Merge k Sorted Lists

带头结点。 /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode(int x) { val = x; }* }*/public class Solution {public ListNode mergeKLists

寻找身高相近的小朋友

题目描述: 小明今年升学到小学一年级,来到新班级后发现其他小朋友们身高参差不齐,然后就想基于各小朋友和自己的身高差对他们进行排序,请帮他实现排序。 输入描述: 第一行为正整数H和N,0<H<200,为小明的身高,0<N<50,为新班级其他小朋友个数。第二行为N个正整数H1-HN,分别是其他小朋友的身高,取值范围0<Hi<200(1<=i<=N),且N个正整数各不相同。 输出描述: 输出

C++ | Leetcode C++题解之第393题UTF-8编码验证

题目: 题解: class Solution {public:static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num &

两个月冲刺软考——访问位与修改位的题型(淘汰哪一页);内聚的类型;关于码制的知识点;地址映射的相关内容

1.访问位与修改位的题型(淘汰哪一页) 访问位:为1时表示在内存期间被访问过,为0时表示未被访问;修改位:为1时表示该页面自从被装入内存后被修改过,为0时表示未修改过。 置换页面时,最先置换访问位和修改位为00的,其次是01(没被访问但被修改过)的,之后是10(被访问了但没被修改过),最后是11。 2.内聚的类型 功能内聚:完成一个单一功能,各个部分协同工作,缺一不可。 顺序内聚:

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟)

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟) 题目描述 给定一个链表,链表中的每个节点代表一个整数。链表中的整数由 0 分隔开,表示不同的区间。链表的开始和结束节点的值都为 0。任务是将每两个相邻的 0 之间的所有节点合并成一个节点,新节点的值为原区间内所有节点值的和。合并后,需要移除所有的 0,并返回修改后的链表头节点。 思路分析 初始化:创建一个虚拟头节点

C语言 | Leetcode C语言题解之第393题UTF-8编码验证

题目: 题解: static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num & MASK1) == 0) {return