摘要:題目解答方法一由于兩個(gè)數(shù)組都是排好序的,因此首先可以想到的思路就是利用歸并排序把兩個(gè)數(shù)組合并成一個(gè)有序的長(zhǎng)數(shù)組,然后直接取出中位數(shù)即可。如果滿(mǎn)足條件,則直接返回求取中位數(shù)。如果減小,則增加,左邊序列數(shù)組的值會(huì)更小,右邊序列數(shù)組的值會(huì)更大。
1. 題目 2. 解答
由于兩個(gè)數(shù)組都是排好序的,因此首先可以想到的思路就是利用歸并排序把兩個(gè)數(shù)組合并成一個(gè)有序的長(zhǎng)數(shù)組,然后直接取出中位數(shù)即可。
class Solution: def findMedianSortedArrays(self, nums1, nums2): """ :type nums1: List[int] :type nums2: List[int] :rtype: float """ len1 = len(nums1) len2 = len(nums2) nums = [] i = 0 j = 0 while (i < len1 and j < len2): if (nums1[i] <= nums2[j]): nums.append(nums1[i]) i += 1 else: nums.append(nums2[j]) j += 1 if (i < len1): while(i < len1): nums.append(nums1[i]) i += 1 if (j < len2): while(j < len2): nums.append(nums2[j]) j += 1 odd = (len1 + len2) % 2 # 長(zhǎng)度是否為奇數(shù) mid = (len1 + len2 - 1) // 2 if (odd): return nums[mid] else: return (nums[mid] + nums[mid+1]) / 2
因?yàn)橐闅v兩個(gè)數(shù)組,所以時(shí)間復(fù)雜度為 $O(m+n)$,而題目中要求算法的時(shí)間復(fù)雜度為 $O(log (m+n))$,因此應(yīng)該是有更高效的算法,借助于二分查找。
所謂中位數(shù),就是將 K 個(gè)數(shù)據(jù)分為長(zhǎng)度相等的兩組,左邊的數(shù)據(jù)小于等于右邊的數(shù)據(jù)。
如果我們要在任意位置 $i$ 處將長(zhǎng)度為 $m$ 的數(shù)組 $A$ 分為兩部分,則共有 $m+1$ 種分法,$i=[0, m]$。
$$ left\_part quad | quad right\_part$$
$$ A[0], A[1], ..., A[i-1] quad | quad A[i], A[i+1], ..., A[m-1]$$
$i=0$ 說(shuō)明左半部分沒(méi)有元素,反之 $i=m$ 說(shuō)明右半部分沒(méi)有元素。左半邊元素個(gè)數(shù)為 $i$ ,右半邊元素個(gè)數(shù)為$m-i$。
同理,我們可以在任意位置 $j$ 處將長(zhǎng)度為 $n$ 的數(shù)組 $B$ 分為兩部分,則共有 $n+1$ 種分法,$j=[0, n]$。
$$ B[0], B[1], ..., B[j-1] quad | quad B[j], B[j+1], ..., B[n-1]$$
針對(duì)劃分完后的數(shù)組 $A$ 和 $B$,如果滿(mǎn)足:
左邊部分的長(zhǎng)度等于右邊部分的長(zhǎng)度(偶數(shù)情況),$i+j = m-i + n-j$;或者等于右邊部分的長(zhǎng)度加 1(奇數(shù)情況) ,$i+j = m-i + n-j+1$。
左邊的最大元素小于等于右邊的最小元素,$A[i-1] <= B[j] quad and quad B[i-1] <= A[j]$。
那我們也就找到了中位數(shù),$median = frac{max(left\_part) + min(right\_part)}{2}$。
所以,我們要做的就是在 $i=[0, m]$ 區(qū)間搜索一個(gè) $i$ 值,使得上面的條件得到滿(mǎn)足。也即
$$ A[i-1] <= B[j] quad and quad B[i-1] <= A[j] ,其中 quad j = frac{m+n+[1]}{2}-i$$
加不加 1 取決于總的長(zhǎng)度是奇數(shù)還是偶數(shù),同時(shí),為了保證 $j$ 的范圍在 $[0, n]$,我們必須要求 $n <= m$。
接下來(lái),我們就在 $i=[0, m]$ 區(qū)間進(jìn)行二分查找。
如果滿(mǎn)足條件,則直接返回求取中位數(shù)。
如果 $i > 0 quad and quad A[i-1] > B[j]$,則減小 $i$。如果增加 $i$,則 $j$ 減小,左邊序列數(shù)組 $A$ 的值會(huì)更大,右邊序列數(shù)組 $B$ 的值會(huì)更小。
如果 $i < m quad and quad B[i-1] > A[j]$,則增加 $i$。如果減小 $i$,則 $j$ 增加,左邊序列數(shù)組 $A$ 的值會(huì)更小,右邊序列數(shù)組 $B$ 的值會(huì)更大。
最后,我們求得左半部分的最大值以及右半部分的最小值,然后就可以求出中位數(shù)了。
因?yàn)椋檎业姆秶鸀?$i=[0, m]$,而且每次查找縮小區(qū)間為原來(lái)的一半,因此時(shí)間復(fù)雜度為 $O(log(min(m, n))$,空間復(fù)雜度為 $O(1)$。
class Solution { public: double findMedianSortedArrays(vector& nums1, vector & nums2) { int m = nums1.size(); int n = nums2.size(); int len = m + n; int odd = len % 2; int left = 0; int i = 0, j = 0; vector ::iterator A = nums1.begin(); vector ::iterator B = nums2.begin(); // 確保數(shù)組 A 的長(zhǎng)度小于等于數(shù)組 B 的長(zhǎng)度 if (m > n) { int temp = m; m = n; n = temp; A = nums2.begin(); B = nums1.begin(); } int right = m; while(left <= right) { i = left + (right - left) / 2; j = (len + odd) / 2 - i; if (i > 0 && A[i-1] > B[j]) { right = i - 1; } else if(i < m && B[j-1] > A[i]) { left = i + 1; } else { break; } } int left_max = 0; int right_min = 0; // 左半部分的最大值 if (i == 0) left_max = B[j-1]; else if (j == 0) left_max = A[i-1]; else left_max = A[i-1] <= B[j-1] ? B[j-1] : A[i-1]; // 右半部分的最小值 if (i == m) right_min = B[j]; else if (j == n) right_min = A[i]; else right_min = A[i] <= B[j] ? A[i] : B[j]; if (odd) { return left_max; } else { return float(left_max + right_min) / 2; } } };
獲取更多精彩,請(qǐng)關(guān)注「seniusen」!
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/44957.html
摘要:解法解法應(yīng)該是最常見(jiàn)的一種解法,就是將兩個(gè)數(shù)組頭尾相加合并起來(lái),然后重新排序,例如輸入兩個(gè)數(shù)組,合并之后變?yōu)椋缓笄蟪鲋形粩?shù)。如果兩個(gè)數(shù)組長(zhǎng)度和為偶數(shù),那么中位數(shù)有兩個(gè),否則只有一個(gè) 題目 There are two sorted arrays nums1 and nums2 of size m and n respectively. Find the median of the tw...
摘要:難度為這個(gè)題目描述很清晰給出兩個(gè)排序好的數(shù)組求這兩個(gè)數(shù)組的中位數(shù)在解這個(gè)題的過(guò)程中會(huì)碰到以下的問(wèn)題先合起來(lái)重新排序是不可行的時(shí)間復(fù)雜度太高為先歸并排序也是不可行的時(shí)間復(fù)雜度為用類(lèi)似桶排的方法時(shí)間復(fù)雜度為不可行可能會(huì)碰到多種全部大于或全部小于 There are two sorted arrays nums1 and nums2 of size m and n respectively...
摘要:分布式的管理和當(dāng)我在談?wù)摷軜?gòu)時(shí)我在談啥狀態(tài)碼詳解無(wú)狀態(tài)協(xié)議和請(qǐng)求支持哪些方法分層協(xié)議棧有哪些數(shù)據(jù)結(jié)構(gòu)運(yùn)用場(chǎng)景說(shuō)說(shuō)你常用的命令為什么要有包裝類(lèi)面向?qū)ο蟮奶卣魇巧妒巧队惺裁春锰幭到y(tǒng)設(shè)計(jì)工程在線(xiàn)診斷系統(tǒng)設(shè)計(jì)與實(shí)現(xiàn)索引背后的數(shù)據(jù)結(jié)構(gòu)及算法原理軟技能 HTTP 【HTTP】分布式session的管理 【HTTP】Cookie和Session 【HTTP】當(dāng)我在談?wù)揜estFul架構(gòu)時(shí)我在談啥?...
摘要:先去空白,去掉空白之后取第一個(gè)字符,判斷正負(fù)符號(hào),若是英文直接返回,若數(shù)字則不取。回文數(shù)題目描述判斷一個(gè)整數(shù)是否是回文數(shù)。回文數(shù)是指正序從左向右和倒序從右向左讀都是一樣的整數(shù)。 JS算法題之leetcode(1~10) 前言 一直以來(lái),前端開(kāi)發(fā)的知識(shí)儲(chǔ)備在數(shù)據(jù)結(jié)構(gòu)以及算法層面是有所暫缺的,可能歸根于我們的前端開(kāi)發(fā)的業(yè)務(wù)性質(zhì),但是我認(rèn)為任何的編程崗位都離不開(kāi)數(shù)據(jù)結(jié)構(gòu)以及算法。因此,我作為...
摘要:最新解法及思路有兩個(gè)有序數(shù)組和,他們的大小各是和,請(qǐng)找出這兩個(gè)數(shù)組所有數(shù)的中位數(shù),總得時(shí)間復(fù)雜度不超過(guò)歸并計(jì)數(shù)法復(fù)雜度時(shí)間空間思路如果對(duì)時(shí)間復(fù)雜度沒(méi)有要求,這個(gè)方法是實(shí)現(xiàn)起來(lái)最簡(jiǎn)單的,我們只需要從下往上依次數(shù)個(gè)元素即可。 Median of Two Sorted Arrays 最新解法及思路:https://yanjia.li/zh/2018/11/... There are two...
閱讀 522·2023-04-26 00:33
閱讀 3547·2021-11-24 09:39
閱讀 2948·2021-09-22 15:34
閱讀 2324·2019-08-23 18:07
閱讀 2918·2019-08-23 18:04
閱讀 3707·2019-08-23 16:06
閱讀 2900·2019-08-23 15:27
閱讀 1620·2019-08-23 14:32