Leetcode
int binarySort(int A[], int lo, int hi, int target){while(lo <= hi){int mid = lo + (hi - lo)/2;if(A[mid] == target)return mid;if(A[mid] < target)lo = mid + 1;elsehi = mid - 1;}}
eg. [7, 8, 9, 3, 4, 5, 6]
在數組中有且僅有一個 斷點 (數字由大變小)。
還是通過折半來實現,關鍵是如有巧妙的繞過這個斷點。
1.如果前半部分有序:
target 在該區間內,則 hi = mid - 1
不在該區間內,則 lo = mid + 1
2.如果前半部分無序
target 在後半部分的有序區間內,則 lo = mid + 1
不在該區間內, 則 hi = mid - 1
也就是說,我們優先考慮有序的部分。
int search(int A[], int lo, int hi, int target){while (lo <= hi){int mid = (lo + hi) / 2;if (A[mid] == target)return mid;if (A[lo] <= A[mid]){if (A[lo] <= target && target < A[mid])hi = mid - 1;elselo = mid + 1;} else {if (A[mid] < target && target <= A[hi-1])lo = mid + 1;elsehi = mid - 1;}}return -1;}
允許重復元素,則上一題中如果A[m]>=A[l],那麼[l,m]為遞增序列的假設就不能成立了,比如[1,3,1,1,1]。
如果A[m]>=A[l]不能確定遞增,那就把它拆分成兩個條件:
若A[m]>A[l],則區間[l,m]一定遞增
若A[m]==A[l]確定不了,那就l++,往下看一步即可。
bool search(int A[], int lo, int hi, int target){while (lo <= hi){int mid = (lo + hi) / 2;if (A[mid] == target)return true;if (A[lo] < A[mid]){if (A[lo] <= target && target < A[mid])hi = mid - 1;elselo = mid + 1;} else if(A[lo] > A[mid]) {if (A[mid] < target && target <= A[hi-1])lo = mid + 1;elsehi = mid - 1;}elselo++;}return false;}