将数组划分为左侧,右侧两部分,分别求左侧和右侧的最大元素和次大元素,全局最大元素是左侧最大元素和右侧最大元素的更大者。
如果左侧的最大元素更大,则全局次大元素来自于右侧最大元素和左侧次大元素的更大者;
如果右侧的最大元素更大,则全局次大元素来自于左侧最大元素和右侧次大元素的更大者。
class solution {
public:
void max2(int A[], int lo, int hi, int &x1, int &x2) {
if (lo + 2 == hi){
if (A[x1 = lo] < A[x2 = hi - 1])
swap(x1, x2);
return;
}//T(2) = 1
if (lo + 3 == hi) {
int mid = (lo + hi) / 2;
if ((A[lo] > A[mid]) && (A[lo] > A[hi - 1])){
x1 = A[lo];
x2 = (A[mid] > A[hi - 1]) ? A[mid] : A[hi - 1];
}
else if ((A[mid] > A[lo]) && (A[mid] > A[hi - 1])) {
x1 = A[mid];
x2 = (A[lo] > A[hi - 1]) ? A[lo] : A[hi - 1];
}
else {
x1 = A[hi - 1];
x2 = (A[lo] > A[mid]) ? A[lo] : A[mid];
}
return;
}//T(3) <= 3
int mi = (lo + hi) / 2;
int x1L, x2L; max2(A, lo, mi, x1L, x2L);
int x1R, x2R; max2(A, mi, hi, x1R, x2R);
if (A[x1L] > A[x1R]) {
x1 = x1L; x2 = (A[x2L] > A[x1R]) ? x2L : x1R;
}
else {
x1 = x1R; x2 = (A[x1L] > A[x2R]) ? x1L : x2R;
}// 2
}
};//T(n) = 2 * T(n/2) + 2;
算法复杂度:
最差的情况,每次递归需要进行两次比较
T(n) = 2 * T(n/2) + 2 <=5n/3 - 2