【递归打卡1】在两个长度相等的排序数组中找到上中位数

【题目】

给定两个有序数组arr1和arr2,已知两个数组的长度都为N,求两个数组中所有数的上中位数。要求时间复杂度O(logN),空间复杂度O(1)

【举例】

例如 arr1 = [1, 2,3,4],arr2 = [3,4,5,6]。

总共8个数,则中位数就是第 4 小的数,为 3.

例如 arr1 = [0,1,2],arr2 = [3,4,5]。

总共6个数,则中位数就是第 3 小的数,为 2.

【难度】

解答

这道题可以采用递归来解决,注意,这道题数组是有序的,所以它有如下特点:

(1)、当 两个数组的长度为偶数时:

我来举个例子说明他拥有的特点吧。我们假定
arr1 = [1, 2,3,4],arr2 = [3,4,5,6]。则数组的长度为 n = 4。

分别选出这两个数组的上中位数的下标,即

mid1 = (n-1)/2 = 1。

mid2 = (n - 1)/2 = 1。

假如 arr2[mid2] > arr2[mid1],那么我们要找的目标数是一定存在于 arr1[mid1+1...n] 和 arr2[0...mid2]中。而不可能存在于 arr1[0...mid1] 和 arr2[mid2+1...n] 之中。

也就是说,我们接下来只需要在arr1[mid1+1...n] 和 arr2[0...mid2] 中查找就行了。

(2)、当两个数组的长度为奇数时:

假定 arr1 = [1, 2,3,4,5],arr2 = [3,4,5,6,7]。则数组的长度为 n = 5。

mid1 = (n-1)/2 = 2。

mid2 = (n - 1)/2 = 2。

这个时候如果 arr2[mid2] > arr1[mid1] 时,则和上面那个情况有点小差别,这时候目标数只存在于 arr1[mid1...n] 和 arr2[0...mid2]中。注意他们的差别,从arr1[mid1+1...n] => arr1[mid1...n]。

理解了这个原理,配合上代码会更好理解,代码如下:

    public static int getUpMedian(int[] arr1, int[] arr2) {
        if(arr1 == null || arr2 == null )
            return -1;
        // 开始寻找
        return find(arr1, 0, arr1.length - 1, arr2, 0, arr2.length - 1);
    }

    public static int find(int[] arr1, int l1, int r1, int[] arr2, int l2, int r2) {
        int mid1 = l1 + (r1 - l1) / 2;
        int mid2 = l2 + (r2 - l2) / 2;
        // 表示数组只剩下一个数,把两个数组中较小的数返回去
        if (l1 >= r1) {
            return Math.min(arr1[l1], arr2[l2]);
        }
        // 元素个数为奇数,则offset为0,为偶数则 offset 为 1
        int offset = ((r1 - l1 + 1) & 1) ^ 1;// 用位运算比较快
        if (arr1[mid1] < arr2[mid2]) {
            return find(arr1, mid1+offset, r1, arr2, l2, mid2);
        } else if (arr1[mid1] > arr2[mid2]) {
            return find(arr1, l1, mid1, arr2, mid2 + offset, r2);
        } else {
            return arr1[mid1];// 返回 arr2[mid2]也可以。
        }
    }

也可以用迭代来做,反而更加简单,迭代版本如下:

    // 迭代版本
    public int getUpMedian2(int[] arr1, int[] arr2) {
        if (arr1 == null || arr2 == null) {
            return -1;
        }
        int l1 = 0;
        int r1 = arr1.length - 1;
        int l2 = 0;
        int r2 = arr2.length - 1;
        int mid1 = 0;
        int mid2 = 0;
        int offset = 0;
        while (l1 < r1) {
            mid1 = l1 + (r1 - l1) / 2;
            mid2 = l2 + (r2 - l2) / 2;
            offset = ((r1 - l1 + 1) & 1)^1;
            if (arr1[mid1] < arr2[mid2]) {
                l1 = mid1 + offset;
                r2 = mid2;
            } else if (arr1[mid1] > arr2[mid2]) {
                r1 = mid1;
                l2 = mid2 + offset;
            } else {
                return arr2[mid1];
            }
        }
        return Math.min(arr1[l1], arr2[l2]);
    }

原文地址:https://www.cnblogs.com/kubidemanong/p/10562292.html

时间: 2024-11-05 18:50:16

【递归打卡1】在两个长度相等的排序数组中找到上中位数的相关文章

【转】在两个长度相等的排序数组中找到上中位数

[题目] 给定两个有序数组arr1和arr2,已知两个数组的长度都为N,求两个数组中所有数的上中位数.要求时间复杂度O(logN),空间复杂度O(1) [举例] 例如 arr1 = [1, 2,3,4],arr2 = [3,4,5,6]. 总共8个数,则中位数就是第 4 小的数,为 3. 例如 arr1 = [0,1,2],arr2 = [3,4,5]. 总共6个数,则中位数就是第 3 小的数,为 2. [难度] 中 解答 这道题可以采用递归来解决,注意,这道题数组是有序的,所以它有如下特点:

算法总结之 在两个长度相等的排序数组中找到上中位数

题目描述: arr1  和 arr2   长度都为N   求两个数组中所有数的上中位数 要求 时间复杂度 O(logN)  额外空间复杂度O(1) 这道题目的方法比较好玩: 这两个数组如下表示: arr1[start1....end1]     arr2[start2...end2] 如果start1==start2  那么也有start2==end2   此时元素总个数是2个,上中位数为最小的那个 如果start1!=start2  令mid1={start1+end1}/2      mid

【C语言】在两个数成对出现的数组中找到一个单独的数。

//在两个数成对出现的数组中找到一个单独的数.比如{1,2,3.3,1,4.2},即找出4 #include <stdio.h> int find(int arr[], int len) { int i = 0; int ret = 0; for (i = 0; i < len; i++) { ret = ret^arr[i]; } return ret; } int main() { int arr1[] = { 1, 2, 2, 3, 1, 5, 3 }; int arr2[] =

9.27 在两个排序数组中找到第K小的数

[题目]: 给定两个有序数组arr1和arr2,再给定一个整数k,返回所有的数中第K小的数 举例: arr1=[1, 2, 3, 4, 5],arr2=[3, 4, 5],k=1 1是所有数中第1小的数,所以返回1 arr1=[1, 2, 3],arr2=[3, 4, 5, 6],k=4 3是所有数中第4小的数,所以返回3 [要求]: 如果arr1的长度为N,arr2的长度为M,时间复杂度清达到O(log(min{M, N})),额外空间复杂度为O(1) 题目来源:左程云老师<程序员代码面试指南

[LeetCode]4. Median of Two Sorted Arrays两个排序数组合并后的中位数

There are two sorted arrays nums1 and nums2 of size m and n respectively. Find the median of the two sorted arrays. The overall run time complexity should be O(log (m+n)). 解法一:不考虑时间复杂度限制在O(log(m+n)),则将两个数组遍历一遍即可以组合成一个排好序的数组,然后取数组的中位数即可,时间复杂度O(m+n): c

arts打卡 从排序数组中删除重复项

Algorithm 从排序数组中删除重复项 给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度. 不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的条件下完成. 示例 1: 给定数组 nums = [1,1,2], 函数应该返回新的长度 2, 并且原数组 nums 的前两个元素被修改为 1, 2. 你不需要考虑数组中超出新长度后面的元素. 示例 2: 给定 nums = [0,0,1,1,1,2,2,3,3,4], 函数应

一个数组中找到满足和为sum的两个数

如果考虑hashmap直接O(n)的速度, 如果不行,就先排序,两头指针很好推理,关键是 a[beg] +a[end]>sum,意思就是说a[end]太大了,最小的数的都不满足,所以排除a[end] 绝知此事要躬行 #include<iostream>#include<algorithm>using namespace std; bool find(int *a,int sum,int len,int &ans1,int &ans2){    int *beg

数组中找到和为给定值的两个数

problem 167 & 170 from leetcode; https://leetcode.com/problems/two-sum-ii-input-array-is-sorted/ https://leetcode.com/problems/two-sum-iii-data-structure-design/ 先来看简单的167:给定一个排好序的数组,以及一个值,找出数组中相加等于该值的两个数,假设这样的值始终存在: 这个只要依次遍历该数组,假设当前值为x,那么只需要在数组中找到va

js实现从长度为100的数组中随机抽取5个不同值

方法一:这个是我的方法,望大家指正…… <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>从长度为100的数组中随机抽取5个不同值</title> </head> <body> <script> var arr = new Array(100); var flag_a =