Problem
Suppose an array sorted in ascending order is rotated at some pivot unknown to you beforehand. For example, \([0,1,2,4,5,6,7]\) might become \([4,5,6,7,0,1,2]\). You are given a target value to search. If found in the array return its index, otherwise return \(-1\).
Note: You may assume no duplicate exists in the array.
Note: Your algorithm's runtime complexity must be in the order of \(O(log n)\).
Example 1:
Input: nums = [4,5,6,7,0,1,2], target = 0
Output: 4
Example 2:
Input: nums = [4,5,6,7,0,1,2], target = 3
Output: -1
Input
nums
- an array of integerstarget
- integer value to search for
Approach
This is a classic textbook problem with a slight twist. The brute force approach is to linearly traverse the array and return the index of a value that matches target
as soon as you encounter one. This is \(O(n)\), which is good, but we could do better.
Since the array is partially sorted, it turns out that this array is still a good candidate for binary search. The key to solving this is to understand your boundaries thoroughly.
The traditional binary search algorithm gives you access to the lowest index, the highest index, and using the two indices you can determine the middle index. The cycle (where the smallest number is right after the largest number) can only exist in one half of the array: either the left-half, or the right-half. Therefore...
- If \(A[lowest] < A[middle]\), then the left half is sorted normally, and the right half contains the cycle.
- Likewise, if \(A[middle] < A[highest]\), then the right half is sorted normally, and the left half contains the cycle.
Once you know where the cycle is, all you have to do is compare your target
value against what you know and don't know, i.e. if the target
value is not in the sorted half of the array, then it must be in the array where the cycle is.
Solution
def search(self, nums: 'List[int]', target: 'int') -> 'int':
def _find(lo, hi, target):
if lo > hi:
return -1
mid = (lo + hi) // 2
if nums[mid] == target:
return mid
elif nums[lo] < nums[mid] and (target < nums[lo] or target > nums[mid]): # the left half is sorted.. so cycle must be in right half
return _find(mid+1, hi, target)
elif nums[lo] < nums[mid] and nums[lo] <= target < nums[mid]:
return _find(lo, mid-1, target)
elif nums[mid] < nums[hi] and (target < nums[mid] or target > nums[hi]):
return _find(lo, mid-1, target)
else:
return _find(mid+1, hi, target)
return _find(0, len(nums)-1, target)