zoukankan      html  css  js  c++  java
  • LEETCODE 返回两个正序数组的 中位数

    代码来自
    此代码为二分查找,时间复杂度为log2(m+n)

    • [ ] todo 此题第k小解法待完成 求第k小的数即可求个 中位数
    /**
     * @Author: WhaleFall541
     * @Date: 2021/6/8 20:23
     * 给定两个大小分别为 m 和 n 的正序(从小到大)数组 nums1 和 nums2。请你找出并返回这两个正序数组的 中位数
     * @see <a href="https://leetcode-cn.com/problems/median-of-two-sorted-arrays/">https://leetcode-cn.com/problems/median-of-two-sorted-arrays/</a>
     */
    public class FindMedianSortedArrays {
        public static void main(String[] args) {
            int[] a1 = {1};
            int[] a2 = {1, 2, 2, 3};
            double res = findMedianSortedArrays(a1, a2);
            System.out.println("res = " + res);
        }
    
        public static double findMedianSortedArrays(int[] nums1, int[] nums2) {
            if (nums1.length > nums2.length) {
                int[] temp = nums1;
                nums1 = nums2;
                nums2 = temp;
            }
    
            int m = nums1.length;
            int n = nums2.length;
    
            // 分割线左边的所有元素需要满足的个数
            int totalLeft = (m + n + 1) / 2;
    
            // 在 nums1 的区间 [0, m] 里查找恰当的分割线,
            // 使得 nums1[i - 1] <= nums2[j] && nums2[j - 1] <= nums1[i]
            int left = 0;
            // NOTE: 这里的m 如果你当指针写成m-1 当短的数组长度为1时 right = m-1 = 0;
            int right = m;
    
    
            while (left < right) {
                // NOTE: right - left + 1 此处+1是为避免一种情况
                // 当left =1,right =2 时 left + (right - left) / 2=1;
                // 设if条件走了else分支 left又被赋值为1了 然后就没玩没了次循环了
                int i = left + (right - left + 1) / 2;
                int j = totalLeft - i;
                if (nums1[i - 1] > nums2[j]) {
                    // 下一轮搜索的区间 [left, i - 1]
                    right = i - 1;
                } else {
                    // 下一轮搜索的区间 [i, right]
                    left = i;
                }
            }
    
            int i = left;
            int j = totalLeft - i;
    
            // NOTE: i 表示 分界线有右边的第一个元素 当i为0时说明数组左边越界处理为一个极小的值
            int nums1LeftMax = i == 0 ? Integer.MIN_VALUE : nums1[i - 1];
            // NOTE: 角标为m、n时数组已经右边越界 此时处理为一个很大的值
            int nums1RightMin = i == m ? Integer.MAX_VALUE : nums1[i];
            int nums2LeftMax = j == 0 ? Integer.MIN_VALUE : nums2[j - 1];
            int nums2RightMin = j == n ? Integer.MAX_VALUE : nums2[j];
    
            if (((m + n) % 2) == 1) {
                return Math.max(nums1LeftMax, nums2LeftMax);
            } else {
                return (double) ((Math.max(nums1LeftMax, nums2LeftMax) + Math.min(nums1RightMin, nums2RightMin))) / 2;
            }
        }
    }
    

    参考资料

    1. LEETCODE视频地址

    转载请注明 原文地址

  • 相关阅读:
    【TopCoder
    【TopCoder
    min_25筛学习笔记
    min_25筛学习笔记
    【SPOJ】 —DIVCNTK(min_25筛)
    【SPOJ】 —DIVCNTK(min_25筛)
    【UOJ#13 188】—Sanrd(min_25筛)
    【UOJ#13 188】—Sanrd(min_25筛)
    【BZOJ5244】【FJWC2018】—最大真因数(min_25筛)
    【BZOJ5244】【FJWC2018】—最大真因数(min_25筛)
  • 原文地址:https://www.cnblogs.com/whalefall541/p/14869189.html
Copyright © 2011-2022 走看看