假设按照升序排序的数组在预先未知的某个点上进行了旋转。
( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] )。
搜索一个给定的目标值,如果数组中存在这个目标值,则返回它的索引,否则返回 -1 。
你可以假设数组中不存在重复的元素。
你的算法时间复杂度必须是 O(log n) 级别。
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/search-in-rotated-sorted-array
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
如果数组「未旋转」,在数组中查找一个特定元素 target 的过程为:
若 target == nums[mid],直接返回
若 target < nums[mid],则 target 位于左侧区间 [left,mid) 中。令 right = mid-1,在左侧区间查找
若 target > nums[mid],则 target 位于右侧区间 (mid,right] 中。令 left = mid+1,在右侧区间查找
但是这道题,由于数组「被旋转」,所以左侧或者右侧区间不一定是连续的。在这种情况下,如何判断 target 位于哪个区间?
根据旋转数组的特性,当元素不重复时,如果 nums[i] <= nums[j],说明区间 [i,j] 是「连续递增」的。
i、j 可以重合,所以这里使用的比较运算符是「小于等于」
因此,在旋转排序数组中查找一个特定元素时:
若 target == nums[mid],直接返回
若 nums[left] <= nums[mid],说明左侧区间 [left,mid]「连续递增」。此时:
若 nums[left] <= target <= nums[mid],说明 target 位于左侧。令 right = mid-1,在左侧区间查找
否则,令 left = mid+1,在右侧区间查找
否则,说明右侧区间 [mid,right]「连续递增」。此时:
若 nums[mid] <= target <= nums[right],说明 target 位于右侧区间。令 left = mid+1,在右侧区间查找
否则,令 right = mid-1,在左侧区间查找
注意:区间收缩时不包含 mid,也就是说,实际收缩后的区间是 [left,mid) 或者 (mid,right]
作者:imageslr
链接:https://leetcode-cn.com/problems/search-in-rotated-sorted-array/solution/yi-wen-jie-jue-4-dao-sou-suo-xuan-zhuan-pai-xu-s-2/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
class Solution {
public static int search(int[] nums, int target) {
int lo=0, hi=nums.length-1, mid=0;
while(lo <= hi){
mid = lo + (hi-lo)/2;
if(nums[mid]==target) return mid;
if(nums[lo]<= nums[mid]){
if(nums[lo]<=target && target<=nums[mid]){
hi = mid-1;
}else {
lo = mid+1;
}
}else{
if(nums[mid]<=target && target <= nums[hi]){
lo = mid+1;
}else{
hi = mid-1;
}
}
}
return -1;
}
}