LeetCode HOT 100 —— 4.寻找两个正序数组的中位数

news2024/10/6 12:31:42

题目

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

算法的时间复杂度应该为 O(log (m+n))
在这里插入图片描述
在这里插入图片描述

思路

正序数组,立即推—>二分查找

如果本题不要求时间复杂度O(log(n + m )),可以考虑暴力解法,时间复杂度为O(n + m)

方法一:暴力

先将两个数组合并,两个有序数组的合并也是归并排序中的一部分,然后根据奇数还是偶数,返回中位数

java代码如下:

class Solution {
	public int findMedianSortedArrays(int[] nums1,int[] nums2){
		int[] nums;//创建一个新数组
		int m = nums1.length;
		int n = nums2.length;
		nums = new int [m + n];//长度为两个数组之和
		//若有一个数组为空,那么直接返回另一个数组的中位数
		if(m == 0){//若nums1为空
			if(n % 2 == 0){
				return (nums2[n / 2 - 1] + nums2[n / 2]) / 2.0;
			} else {
				return nums2[n / 2];
			}
		}	
		if(n == 0){//若nums2为空
			if(m % 2 == 0){
				return (nums1[m / 2 - 1] + nums1[m / 2]) / 2.0;
			} else {
				return nums1[m / 2];
			}
		}
		//若两个数组均不为空,则加入到新数组
		int count = 0;//新数组下标,记录新数组的长度
		int i = 0, j = 0;
		while(count != (m + n)){
			if(i == m){//如果nums1已经遍历完了
				while(j != n){
					nums[count++] = nums2[j++];
				}
				break;//当两个数组都遍历完时,退出循环
			}
			if(j == n){
				while(i != m){
					nums[count++] = nums1[i++]; 
				}
				break;
			}
			
			if(nums1[i] < nums2[j]){
				nums[count++] = nums1[i++];
			} else {
				nums[count++] = nums2[j++];
			}
		}
					
		//求新数组的中位数
		if(count % 2 == 0){
			return ((nums[count / 2 - 1] + nums[count / 2 ]) / 2.0);
		} else {
			return nums[count / 2];
		}
	}
}

另一种思路是不用合并两个有序数组,只用找到中位数的位置即可,因为两个数组长度知道,那么合并之后的数组的中位数的下标位置也可以计算出来,然后用两个指针,分别指向两个数组,每次将较小值的指针后移一位,直到到达中位数位置,这种方法虽然空间复杂度降为了O(1),时间复杂度仍然是O(n + m)

方法二:二分查找

题目中的log的时间复杂度,只有用二分的方法才能达到

根据中位数定义,当m + n是奇数时,中位数是两个有序数组中的第(m+n)/2个元素,当m+n是偶数时,中位数是两个有序数组中的第(m+n)/2个元素和(m+n)/2 +1个元素的平均值(注意,这里说的是第几个元素,下标是需要在减一的),因此本题可以转化成寻找两个有序数组中的第 k 小的数,其中k为(m+n)/2或者(m+n)/2 + 1

假设两个有序数组分别是 A 和 B。要找到第 k 个元素,可以比较 A[k/2−1]B[k/2−1],由于 A[k/2−1]B[k/2−1]的前面分别有 A[0 .. k/2−2]B[0 .. k/2−2],即 k/2−1个元素,对于 A[k/2−1]B[k/2−1] 中的较小值,最多只会有 (k/2−1)+(k/2−1)≤k−2个元素比它小,那么它就不能是第 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个数,即比 A[k/2−1]小的数最多只有 k−2个,因此 A[k/2−1]不可能是第 k 个数,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/2不可能是第 k 小的数,查找范围缩小了一半(比如排除了A[0]A[k/2−1],一共k/2个元素)。

同时,在排除后的新数组上继续进行二分查找,并且根据我们排除数的个数,减少 k 的值,这是因为排除的数都不大于第 k 小的数。

有以下三种情况需要特殊处理:

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

java代码如下:

class Solution {
	public double findMedianSortedArrays(int[] nums1,int[] nums2){
		int m = nums1.length;
		int n = nums2.length;
		int totalLen = m + n;
		if(totalLen % 2 == 1){
			int midIdx = totalLen / 2;//这里是数组的下标
			double median = getKThNumber(nums1,nums2,midIdx + 1);//这里参数是第k小的数,所以数组下标midIdx对应的就是第midIdx + 1小的数
			return median;
		} else {//如果数组长度是偶数
			int midIdx1 = totalLen / 2 - 1;
			int midIdx2 = totalLen / 2;//分别表示数组下标
			double median = (getKThNumber(nums1,nums2,midIdx1 + 1) + getKThNumber(nums1,nums2,midIdx2 + 1)) / 2.0;
			return median; 
		}
	}
	
	public int getKThNumber(int[] nums1,int[] nums2,int k){
	
		int m = nums1.length;
		int n = nums2.length;
		int idx1 = 0, idx2 = 0;
		int kthNumber = 0;
		
		while(true){
			//处理边界情况
			if(idx1 == m){//如果数组为空,即长度为0,那么直接返回另一个数组第k小的元素
				return nums2[idx2 + k -1];
			}
			if(idx2 == n){
				return nums1[idx1 + k - 1];
			}
			if(k == 1){//则返回两个数组首元素的最小值即可
				return Math.min(nums1[idx1],nums2[idx2]);
			}
			
			//正常情况
			int half = k / 2;//每次都能排除k/2个元素
			int newIdx1 = Math.min(idx1 + half, m) - 1;
			int newIdx2 = Math.min(idx2 + half, n) - 1;
			int pivot1 = nums1[newIdx1], pivot2 = nums2[newIdx2];
			if(pivot1 <= pivot2){
				k -= (newIdx1 - idx1 + 1);//k减去排除的元素数量
				idx1 = newIdx1 + 1;
			} else {
				k -= (newIdx2 - idx2 + 1);
				idx2 = newIdx2 + 1;
			}
		}
	}
}

时间复杂度:每进行一次循环,就减少 k/2 个元素,所以时间复杂度是 O(log(k),而 k=(m+n)/2,所以最终的复杂也就是 O(log(m+n)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/28281.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

光点数据可视化解决方案,助力新型智慧城市打造_光点科技

随着城市化进程的快速发展&#xff0c;智慧城市逐渐从理论理念演变为实践。智慧城市作为一个极其复杂的城市数字化建设和运营系统&#xff0c;涵盖了大量的智能交通、智能物流、智能公园等子系统。对于智慧城市解决方案提供商和集成商来说&#xff0c;其数据可视化产品的统一监…

View的绘制流程

view的绘制流程主要为measure&#xff0c;layout&#xff0c;draw三个阶段 View与window的逻辑结构如图所示&#xff1a; ViewRootImpl&#xff08;替代ViewRoot&#xff09;类&#xff0c;是连接WindowMannager和DecorView的纽带&#xff0c;View的三大流程均是通过ViewRoot完…

godoc安装与go文档查询

前言 最近在用go语言做项目&#xff0c;语法还是很简单的&#xff0c;但是API不熟悉&#xff0c;&#x1f605;&#xff0c;另外也没有类似Java的Spring这样的独秀的IOC和AOP框架&#xff0c;灵活度很大&#xff0c;经常需要查询文档&#xff0c;但是godoc从go1.14.2开始就从内…

【CNN】MobileNet——卷积神经网络中轻量级的经典

前言 MobileNet 系列 是 Andrew G. Howard&#xff08;Google Inc.&#xff09; 等人于 2017 年&#xff08;其实是 2016 年先于 Xception 已经提出&#xff0c;但是直到 2017 年才挂到 arXiv 上&#xff09;在 MobileNets: Efficient Convolutional Neural Networks for Mobi…

干货 | 如何在子查询和联接之间选择

在 联接与子查询&#xff1a;哪个更快&#xff1f;文章我们了解到连接往往比子查询执行得更快。话虽如此&#xff0c;这并不是一条通用准则&#xff0c;因此你可能不希望自动假设联接更可取。正如那篇文章中提到的&#xff0c;如果你需要在查询添加许多联接&#xff0c;数据库服…

分治算法Divide and Conquer

评价 它可以减少运行的时间&#xff0c;很多问题如果暴力求解需要O(n^2)的复杂度&#xff0c;而通过分治可以减少到O&#xff08;nlogn&#xff09; 当与随机化技术相结合时&#xff0c;分治的功能很强大 分治算法的步骤 1.先将大的问题分解为一个个小的子问题 2.对每一个子…

swift内存绑定

swift提供了3种不同的API来绑定/重新绑定指针 assumingMemoryBound(to:)bindMemory(to: capacity:)withMemoryRebound(to: capacity: body:) 绕过编译器检查 - assumingMemoryBound 就是假定内存绑定 func testPointer(_ p: UnsafePointer<Int>) {print(p) } let tup…

Tomcat运行流程、Servlet运行原理以及常用API

文章目录Servlet原理Tomcat 的定位Tomcat 的伪代码Tomcat 初始化流程Tomcat处理请求总结Servlet的核心APIHttpServletHttpServletRequestHttpServletResponseCookie 和 SessionServlet原理 Servlet终究是属于应用层&#xff0c;它是在应用层进行的一系列操作&#xff0c;它的底…

Prometheus Operator 实战 监控 etcd 集群

上节课和大家讲解了 Prometheus Operator 的安装和基本使用方法&#xff0c;这节课给大家介绍如何在 Prometheus Operator 中添加一个自定义的监控项。 除了 Kubernetes 集群中的一些资源对象、节点以及组件需要监控&#xff0c;有的时候我们可能还需要根据实际的业务需求去添…

Java 后端 本地调试-获取微信公众号 openId

Java 后端 本地调试-获取微信公众号 openId申请测试微信公众号内网穿透工具配置公众号获取用户 openId申请测试微信公众号 微信测试公众号 内网穿透工具 netapp 配置公众号 搜索网页账号选项 点击修改&#xff0c;填写内网穿透的域名 获取用户 openId 1 第一步&#xff…

国家高新技术企业的好处

国家高新技术企业的好处&#xff1a;享受税收减免优惠政策&#xff1b;国家科研经费支持和财政拨款&#xff1b;国家级的资质认证硬招牌&#xff1b;提升企业品牌形象&#xff1b;促进企业科技转型&#xff1b;提高企业市场价值&#xff1b;提高企业资本价值&#xff1b;吸引市…

【电脑讲解】电脑如何实现双系统

核心提示&#xff1a;电脑双系统&#xff0c;大家应该不会太陌生&#xff0c;有的网吧就装的是双系统&#xff0c;双系统可以满足不同人群的需要&#xff0c;可以这样说&#xff0c;一个系统可以专门工作使用&#xff0c;另一个可以供玩游戏使用&#xff0c;&#xff08;电脑硬…

[Linux打怪升级之路]-环境变量

前言 作者&#xff1a;小蜗牛向前冲 名言&#xff1a;我可以接受失败&#xff0c;但我不能接受放弃 如果觉的博主的文章还不错的话&#xff0c;还请点赞&#xff0c;收藏&#xff0c;关注&#x1f440;支持博主。如果发现有问题的地方欢迎❀大家在评论区指正。 目录 一、认识环…

一文详解JVM的内存结构

目录 前言 内存结构 程序计数器 虚拟机栈 本地方法栈 堆内存 方法区 内部组成 前言 Java的JVM解决的问题是跨操作系统问题。程序员只需要专注于代码的编写&#xff0c;这些代码能够在不同的操作系统Mac&#xff0c;Linux和Windows运行的前提是JVM。JVM还提供了垃圾回收机制…

Linux终端操作-Xshell和Xftp(家庭版)

目录一&#xff0c;终端操作二&#xff0c;软件安装1&#xff0c;Xshell, Xftp下载2&#xff0c;Xshell安装3&#xff0c;Xftp安装三&#xff0c;使用1&#xff0c;Xshell建立连接2&#xff0c;Xftp上传文件一&#xff0c;终端操作 上一篇博客记录了如何本地安装虚拟机并实现本…

SpringSecurity(十五)---OAuth2的运行机制(上)-OAuth2概念和授权码模式讲解

一、前言 鸽了很久&#xff0c;其实也因为自己确实比较忙&#xff0c;加之自己在造demo的时候也遇到了很多问题&#xff0c;并且网上这方面的解答非常之少&#xff0c;不过也正是因为少&#xff0c;才更加让我想写这样的知识分享&#xff0c;最终&#xff0c;在一篇博客的解答…

中文drupal教程(4)Session会话系统

Session&#xff08;会话&#xff09;在网站中扮演非常重要的角色&#xff0c;储存临时用户数据、登录数据等等都用到了它&#xff0c;Drupal使用到了Symfony的Session组件&#xff0c;该组件非常强大灵活&#xff0c;drupal在此基础上有所改造和扩展&#xff0c;要理解Symfony…

企业微信接口测试实战(一)

本文为在霍格沃兹测试开发学社中学习到的一些技术,写出来分享给大家,希望有志同道合的小伙伴可以一起交流技术,一起进步~ 霍格沃茨启发: 测试开发进阶班>接口自动化测试>企业微信接口测试实战 企业微信接口测试实战 一、准备环境二、脚本实现2.1、 获得access_token2…

防火墙用户管理理论+实验

目录 注&#xff1a;实验需要有安全策略配置、NAT配置基础 一、防火墙用户管理重要知识点 用户管理 访问控制策略 NGFW下一代防火墙 AAA 鉴别方式——认证 用户认证的分类&#xff1a; 上网用户上线流程&#xff1a; 二、用户认证实验&#xff1a; 实验拓扑 先配置防…

pmp考试是什么?

PMP是一个全球资格认证&#xff0c;也是目前项目管理领域大家公认的证书&#xff0c;相当于项目管理的入门证书。 一、PMP 是什么 pmp 中文叫项目管理专业人士资格认证&#xff0c;目前项目管理领域大家公认的证书&#xff0c;是一个用来评估项目管理人员的知识技能是否已经达…