1. 程式人生 > >每日一題之 兩個排序好的陣列的中為數

每日一題之 兩個排序好的陣列的中為數

描述:

給定連個排序好的陣列,求這兩個陣列的中為數;
如:
a: 1 2 3
b: 4 5
中位數為 3

思路:

本題若沒有限制時間複雜度為O(log(m+n))的話,對兩個陣列使用歸併排序,很容易可以找到他們的中位數,所用時間複雜度為O(m*n)。但是要將時間複雜度降為O(log(m+n)),就需要嘗試對兩個陣列同時進行二分查詢,逐步排除掉不可能出現中位數的區間,最後找到所求的中位數。這種解法的主要思想就是:
如果陣列a的中位數小於陣列b的中位數,那麼整體的中位數只可能出現在a的右區間加上b的左區間之中;
如果陣列a的中位數大於等於陣列b的中位數,那麼整體的中位數只可能出現在a的左區間加上b的右區間之中。
關鍵就是利用分治的思想逐漸縮小a的區間和b的區間來找到中位數。

#include <bits/stdc++.h>

using namespace std;

double findKth(vector<int>a, int ida, vector<int>b, int idb, int k) {
	if (ida > (int)a.size() - 1)
		return b[idb + k - 1];
	if (idb > (int)b.size()-1)
		return a[ida + k - 1];
	if (k == 1)
		return min(a[ida],b[idb]);

	int mida =
9999; int midb = 9999; if (ida + k/2 - 1 < (int)a.size()) mida = a[ida + k/2 - 1]; if (idb + k/2 - 1 < (int)b.size()) midb = b[idb + k/2 - 1]; if (mida < midb) { return findKth(a,ida + k/2,b,idb,k-k/2); } return findKth(a,ida,b,idb + k/2, k-k/2); } double findMedianSortedArrays(vector<
int>&nums1, vector<int>&nums2) { int m = nums1.size(); int n = nums2.size(); double res = 0; if (m == 0) { if (n%2 != 0) res = 1.0*nums2[n/2]; else res = (nums2[n/2]+nums2[n/2-1])/2.0; return res; } if (n == 0) { if (m%2 != 0) res = nums1[m/2]; else res = (nums1[m/2]+nums1[m/2-1])/2.0; return res; } int total = (m+n+1)/2; int total2 = (m+n+2)/2; res = findKth(nums1,0,nums2,0,total) + findKth(nums1,0,nums2,0,total2); return res/2; } void solve(vector<int>&a, vector<int>&b) { cout << findMedianSortedArrays(a,b) << endl; } int main() { int m = 2; int n = 3; vector<int>a; vector<int>b; int x; for (int i = 0; i < m+n; ++i) { cin >> x; if (i < m) a.push_back(x); else b.push_back(x); } solve(a,b); return 0; }