【力扣 Hot100 | 第三天】4.12(寻找两个正序数组的中位数)

2024-04-12 20:52

本文主要是介绍【力扣 Hot100 | 第三天】4.12(寻找两个正序数组的中位数),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

**【力扣 Hot100 | 第二天】4.11**

文章目录

  • 3.寻找两个正序数组的中位数
    • 3.1题目
    • 3.2解法:暴力(归并排序)
    • 3.3解法:二分法

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

3.1题目

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

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

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

3.2解法:暴力(归并排序)

  • 题目既然给出了两个正序数组,我们可以将这两个正序数组合成为一个正序的数组
  • 判断合成后的数组的长度为奇数还是偶数,进而求出中位数
  • 归并排序的时间复杂度为 O(m+n)、空间复杂度为 O(m+n)
class Solution {public double findMedianSortedArrays(int[] nums1, int[] nums2) {int len1 = nums1.length;int len2 = nums2.length;// 合成的数组int[] res = new int[len1+len2];int index = 0;int left = 0, right = 0;// 归并排序合成部分while(left<len1&&right<len2) {if(nums1[left]<nums2[right]) {res[index++] = nums1[left++];} else {res[index++] = nums2[right++];}}// 判断nums1剩余还是nums2剩余,并将剩余部分加入到合成的数组之后while(left<len1) {res[index++] = nums1[left++];}while(right<len2) {res[index++] = nums2[right++];}// 判断合成后的数组的长度,求出中位数。返回的是double类型if((len1+len2)%2==1) {return (double) res[(len1+len2)/2];} else {return (double) (res[(len1+len2)/2]+res[(len1+len2)/2-1])/2;}}
}

3.3解法:二分法

  1. 暴力解法对于本题来说不满足时间复杂度,要求时间复杂度为O(log(m+n)),所以自然想到了二分法。

  2. 理解:

    1. 两个数组长度为奇数,则中位数即为第 (m+n)/2+1 小 元素;
    2. 两个数组长度为偶数,则中位数即为第 (m+n)/2 小元素 和 第 (m+n)/2+1 小元素;
  3. 归纳:如何求出两个数组中第k小的元素?

    1. 首先需要找到 第一个数组中的k/2位置、第二个数组的k/2位置;

    2. 如果 nums1[k/2] < nums2[k/2] ,则 nums1[k/2]及其前面的元素均不是第k小,所以我们应该从 nums1[k/2+1] 到末尾,以及nums2中查找

      1. why?

      2. 理由:比nums1[k/2]小的数字有 k/2-1个,比nums2[k/2]小的数字有 k/2-1个;

        又又 nums1[k/2] < nums2[k/2]

        即 比nums2[k/2]小的数字最小有 (k/2-1)+(k/2-1)+1= k-1个

        即nums1[k/2]最多是第k-1个数,它及其前面的肯定不是第k个数,所以需要去掉

    3. 如果 nums1[k/2] > nums2[k/2],则 nums2[k/2]及其后面的元素均不是第k小,所以我们应该从 nums1 以及 nums2[k/2+1]到末尾中查找

  4. 例子:

    image-20240412170933717

public double findMedianSortedArrays(int[] nums1, int[] nums2) {int n = nums1.length;int m = nums2.length;int left = (n + m + 1) / 2;int right = (n + m + 2) / 2;//将偶数和奇数的情况合并,如果是奇数,会求两次同样的 k 。return (getKth(nums1, 0, n - 1, nums2, 0, m - 1, left) + getKth(nums1, 0, n - 1, nums2, 0, m - 1, right)) * 0.5;  
}private int getKth(int[] nums1, int start1, int end1, int[] nums2, int start2, int end2, int k) {int len1 = end1 - start1 + 1;int len2 = end2 - start2 + 1;//让 len1 的长度小于 len2,这样就能保证如果有数组空了,一定是 len1 if (len1 > len2) return getKth(nums2, start2, end2, nums1, start1, end1, k);if (len1 == 0) return nums2[start2 + k - 1];if (k == 1) return Math.min(nums1[start1], nums2[start2]);int i = start1 + Math.min(len1, k / 2) - 1;int j = start2 + Math.min(len2, k / 2) - 1;if (nums1[i] > nums2[j]) {return getKth(nums1, start1, end1, nums2, j + 1, end2, k - (j - start2 + 1));}else {return getKth(nums1, i + 1, end1, nums2, start2, end2, k - (i - start1 + 1));}
}

在这里插入图片描述

这篇关于【力扣 Hot100 | 第三天】4.12(寻找两个正序数组的中位数)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

锐捷和腾达哪个好? 两个品牌路由器对比分析

《锐捷和腾达哪个好?两个品牌路由器对比分析》在选择路由器时,Tenda和锐捷都是备受关注的品牌,各自有独特的产品特点和市场定位,选择哪个品牌的路由器更合适,实际上取决于你的具体需求和使用场景,我们从... 在选购路由器时,锐捷和腾达都是市场上备受关注的品牌,但它们的定位和特点却有所不同。锐捷更偏向企业级和专

vue如何监听对象或者数组某个属性的变化详解

《vue如何监听对象或者数组某个属性的变化详解》这篇文章主要给大家介绍了关于vue如何监听对象或者数组某个属性的变化,在Vue.js中可以通过watch监听属性变化并动态修改其他属性的值,watch通... 目录前言用watch监听深度监听使用计算属性watch和计算属性的区别在vue 3中使用watchE

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

寻找身高相近的小朋友

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

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

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

C语言:柔性数组

数组定义 柔性数组 err int arr[0] = {0}; // ERROR 柔性数组 // 常见struct Test{int len;char arr[1024];} // 柔性数组struct Test{int len;char arr[0];}struct Test *t;t = malloc(sizeof(Test) + 11);strcpy(t->arr,

C 语言基础之数组

文章目录 什么是数组数组变量的声明多维数组 什么是数组 数组,顾名思义,就是一组数。 假如班上有 30 个同学,让你编程统计每个人的分数,求最高分、最低分、平均分等。如果不知道数组,你只能这样写代码: int ZhangSan_score = 95;int LiSi_score = 90;......int LiuDong_score = 100;int Zhou

两数之和--力扣1

两数之和 题目思路C++代码 题目 思路 根据题目要求,元素不能重复且不需要排序,我们这里使用哈希表unordered_map。注意题目说了只对应一种答案。 所以我们在循环中,使用目标值减去当前循环的nums[i],得到差值,如果我们在map中能够找到这个差值,就说明存在两个整数的和为目标值。 如果没有找到,就将当前循环的nums[i]以及下标i放入map中,以便后续查