04. 寻找两个正序数组的中位数 Golang实现
题目:
给定两个大小分别为 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 + 3) / 2 = 2.5
思路分析:
我们通过二分查找一个较短的数组的某个切分点,计算出另一个数组对应的切分点,从而找到中位数。
问题分解:
输入:两个已排序的数组 nums1 和 nums2。
目标:找出这两个数组合并后有序数组的中位数。
要求:时间复杂度为 O(log(m + n))。
关键思路:
[1 ] 中位数的定义:
假设一个数组长度为m,
对于奇数长度的合并数组,中位数是正中间的那个数,对应下标为m/2。
对于偶数长度的合并数组,中位数是中间两个数的平均值,对应下标为(m-1)/2,m/2。
下面为了统一不区分奇数偶数,可以使用:(m+1)/2,涉及到一个很关键的点,就是计算机的整除只取整数部分,即整体有几个偶数。如果原本是偶数,那么计算后的下标(m+1)/2,指向第二个数。而保证了第一个数:无论奇数偶数长度,都在这个下标的左侧。从而达到了统一公式的目的。那么合并两个数组,不确定哪个数组是奇数偶数,只需要记录下来分类点,从而可以获取到左右两侧的值。[2 ] 不直接合并数组:
如果简单地合并数组再取中位数,时间复杂度是 O(m + n),不符合题目要求。
我们可以通过二分查找,将问题转换为寻找“两个数组的切分点”,使得切分线左边的数都小于右边的数。[ 3] 二分查找的核心思想:
先选定较短的数组 nums1,假设长度为 m,较长的数组 nums2 长度为 n。
我们在较短的数组 nums1 上进行二分搜索,找到一个合适的切分点 i。
根据 i,可以计算出 nums2 的切分点 j = (m + n + 1) / 2 - i,这样保证总是有 (m + n + 1) / 2 个元素在左边。因为中位数在有序列表中的位置必然是合并后的数组的中间位置![ 4] 找到中位数的条件:
目标是找到两个切分点 i 和 j,使得:
nums1[i-1] ≤ nums2[j]
nums2[j-1] ≤ nums1[i]
也就是说,左边的最大值小于等于右边的最小值。如果这两个条件成立,那么说明切分点是合理的,可以得到最终中位数。
调整切分点:
如果 nums1[i-1] > nums2[j],说明 i 太大了,需要左移。
如果 nums2[j-1] > nums1[i],说明 i 太小了,需要右移。
通过不断调整 i,最终可以找到一个满足条件的切分点。
中位数的计算:
如果 m + n 是奇数,那么中位数就是左边部分的最大值,即 max(nums1[i-1], nums2[j-1])。
如果 m + n 是偶数,那么中位数是左边最大值和右边最小值的平均值,即 (max(nums1[i-1], nums2[j-1]) + min(nums1[i], nums2[j])) / 2。

具体步骤:
[ 1] 初始化:
让 nums1 的长度为 m,nums2 的长度为 n。
保证 nums1 的长度小于等于 nums2,即 m ≤ n。[2 ] 二分查找:
在 nums1 上进行二分搜索。设定 i 为 nums1 的切分点,j 为 nums2 的切分点,满足 j = (m + n + 1) / 2 - i。通过二分调整 i 的位置,直到满足 nums1[i-1] ≤ nums2[j] 且 nums2[j-1] ≤ nums1[i]。
找到切分点:如果找到满足条件的切分点,计算中位数:
如果 m + n 是奇数,中位数就是左半部分的最大值。
如果 m + n 是偶数,中位数是左半部分的最大值与右半部分的最小值的平均值。[3 ] 边界情况:
当 i 或 j 为 0 或 m 时,需要特别处理,因为此时可能没有左边或右边的元素。
边界处理情况详细说明:
nums1Mid == 0 和 nums1Mid == m 的情况:
- nums1Mid == 0 表示 nums1 的左边部分为空,没有左侧元素,此时划分线完全位于 nums1 的最左边。所以左边的最大值(midLeft)只能来自 nums2。
- nums1Mid == m 表示 nums1 的右边部分为空,划分线完全位于 nums1 的最右边。所以右边的最小值(midRight)只能来自 nums2。
nums2Mid == 0 和 nums2Mid == n 的情况:
- nums2Mid == 0 表示 nums2 的左边部分为空,此时 nums2 左侧没有元素,左边的最大值只能从 nums1 中取得。
- nums2Mid == n 表示 nums2 的右边部分为空,右边的最小值只能从 nums1 中取得。
点击查看代码
func findMedianSortedArrays(nums1 []int, nums2 []int) float64 {
if len(nums1) > len(nums2) {
return findMedianSortedArrays(nums2, nums1)
}
m, n := len(nums1), len(nums2)
low, high, k := 0, m, (m+n+1)/2
var nums1Mid, nums2Mid int
for low <= high {
nums1Mid = low + (high - low) / 2
nums2Mid = k - nums1Mid
if nums1Mid > 0 && nums1[nums1Mid-1] > nums2[nums2Mid] {
// nums1Mid is too large, need to decrease
high = nums1Mid - 1
} else if nums1Mid < m && nums1[nums1Mid] < nums2[nums2Mid-1] {
// nums1Mid is too small, need to increase
low = nums1Mid + 1
} else {
// Found the correct partition
break
}
}
// Determine left half's maximum
var midLeft int
if nums1Mid == 0 {
midLeft = nums2[nums2Mid-1]
} else if nums2Mid == 0 {
midLeft = nums1[nums1Mid-1]
} else {
midLeft = max(nums1[nums1Mid-1], nums2[nums2Mid-1])
}
// If total length is odd, return the left side's maximum
if (m+n) % 2 == 1 {
return float64(midLeft)
}
// Determine right half's minimum
var midRight int
if nums1Mid == m {
midRight = nums2[nums2Mid]
} else if nums2Mid == n {
midRight = nums1[nums1Mid]
} else {
midRight = min(nums1[nums1Mid], nums2[nums2Mid])
}
// Return the average of the two middle values
return float64(midLeft + midRight) / 2.0
}
04. 寻找两个正序数组的中位数 Golang实现的更多相关文章
- leetcode-4. 寻找两个正序数组的中位数
leetcode-4. 寻找两个正序数组的中位数. 给定两个大小为 m 和 n 的正序(从小到大)数组 nums1 和 nums2. 请你找出这两个正序数组的中位数,并且要求算法的时间复杂度为 O(l ...
- 微软面试题: LeetCode 4. 寻找两个正序数组的中位数 hard 出现次数:3
题目描述: 给定两个大小为 m 和 n 的正序(从小到大)数组 nums1 和 nums2.请你找出并返回这两个正序数组的中位数. 进阶:你能设计一个时间复杂度为 O(log (m+n)) 的算法解决 ...
- leetcode 刷题(数组篇)4题 寻找两个正序数组的中位数(二分查找)
题目描述 给定两个大小分别为 m 和 n 的正序(从小到大)数组 nums1 和 nums2.请你找出并返回这两个正序数组的 中位数 . 示例 1: 输入:nums1 = [1,3], nums2 = ...
- Leetcode随缘刷题之寻找两个正序数组的中位数
我一上来没读清题,想着这题这么简单,直接就上手写了: package leetcode.day_12_05; import java.util.ArrayList; import java.util. ...
- leetcode 4. Median of Two Sorted Arrays 寻找两个正序数组的中位数(困难)
一.题目大意 标签: 查找 https://leetcode.cn/problems/median-of-two-sorted-arrays 给定两个大小分别为 m 和 n 的正序(从小到大)数组 n ...
- LeetCode刷题4:寻找两个正序数组的中位数
题目: 给定两个大小分别为 m 和 n 的正序(从小到大)数组 nums1 和 nums2.请你找出并返回这两个正序数组的 中位数 . 算法的时间复杂度应该为 O(log (m+n)) . 来源:力扣 ...
- Leetcode4. 寻找两个正序数组的中位数
> 简洁易懂讲清原理,讲不清你来打我~ 输入两个递增数组,输出中位数
原题地址: median-of-two-sorted-arrays 题目描述: 示例 1: 输入:nums1 = [1,3], nums2 = [2] 输出:2.00000 解释:合并数组 = [1, ...
- LeetCode-004-寻找两个正序数组的中位数
寻找两个正序数组的中位数 题目描述:给定两个大小分别为 m 和 n 的正序(从小到大)数组 nums1 和 nums2.请你找出并返回这两个正序数组的 中位数 . 示例说明请见LeetCode官网. ...
随机推荐
- 面向对象VS面向过程
什么是面向对象呢? 对于接触或者熟悉一些编程知识的同学来讲,"面向对象"这个词儿一点儿也不陌生.经常听说XX语言是完全面向对象的编程语言,比如C#.Java这些便是完全面向对象的编 ...
- 一图看懂网易数帆指标平台EasyMetrics
简化数据分析,提升决策速度!EasyMetrics,指标的全生命周期管理平台. 为何EasyMetrics? 集中化管理,降低门槛.开箱即用,提升查询速度. 适合人群? 业务用户.开发者.数据团队,E ...
- 基于 Vagrant 手动部署多个 Redis Server
环境准备 宿主机环境:Windows 10 虚拟机环境:Vagrant + VirtualBox Vagrantfile 配置 首先,我们需要编写一个 Vagrantfile 来定义我们的虚拟机配置. ...
- log4js 的安装设置和实例
1. 安装 yarn add log4js -D 2.设置 const log4js = require('log4js') 3.实例 var log = log4js.getLogger(); // ...
- oeasy 教您玩转linux 之 010209 装酷利器 hollywood
我们来回顾一下 上一部分我们都讲了什么? 屏幕故障风格的软件包bb 可以设置音频 这次装一个酷 下个hollywood软件包 apt show hollywood apt search hollywo ...
- [rCore学习笔记 014]批处理系统
写在前面 本随笔是非常菜的菜鸡写的.如有问题请及时提出. 可以联系:1160712160@qq.com GitHhub:https://github.com/WindDevil (目前啥也没有 本章目 ...
- 微服务:nacos服务注册与发现
服务治理的三个角色: 服务提供者:订阅服务 服务消费者:注册服务 注册中心:记录与监控服务状态,推送服务变更信息.提供者定时发送心跳检测,心跳检测失败,就会向消费者推送变更 提供者通过负载均衡的算法选 ...
- ios的idp/iep证书的生成方法,无苹果电脑
在这个多端开发的年代,出现了很多优秀的开发框架,比如hbuilder和uniapp等等.我们可以使用这些框架来开发APP,假如我们要打包ios的app,则需要一个idp/iep证书. 那么这个证书是如 ...
- 1、Git简介
1.1.概述 Git 是一个开源免费的分布式版本控制系统,用于快速高效地管理各种小型或大型项目的代码. Git 不仅容易学习.占用空间小,而且性能快如闪电. Git 具有廉价的本地分支.方便的暂存区域 ...
- NVIDIA人形机器人AI套件:NVIDIA Isaac Manipulator 和 NVIDIA Isaac Perceptor
IsaacManipulator 为机械臂提供了卓越的灵活性和模块化AI功能,并提供了一系列强大的基础模型和GPU加速库.它提供了高达80倍的路径规划加速,零样本感知提高了效率和吞吐量,使开发者能够实 ...