力扣 33. 搜索旋转排序数组
假设按照升序排序的数组在预先未知的某个点上进行了旋转。
( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] )。
搜索一个给定的目标值,如果数组中存在这个目标值,则返回它的索引,否则返回 -1 。
你可以假设数组中不存在重复的元素。
你的算法时间复杂度必须是 O(log n) 级别。
示例 1:
输入: nums = [4,5,6,7,0,1,2], target = 0
输出: 4
示例 2:
输入: nums = [4,5,6,7,0,1,2], target = 3
输出: -1
来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/search-in-rotated-sorted-array
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
就像官方说的,最直接暴力的方法就是,遍历一遍数组,自然能知道到底有没有这个数和这个数在哪,但是如上面的题中说到
复杂度必须是O(log n) 级别,而这个暴力解法的时间复杂度为O(n),显然不符合条件
还有一种方法就是二分搜索法,这个官方也给出了,我对二分法的运用还不熟练,故写次笔记以便查阅
旋转排序数组,可能刷过剑指offer的大佬可能并不陌生,里面有一道题叫找旋转排序数组中的最小数,那个题我记得
也是用了二分法
首先我们分析一下旋转排序数组的特点,比方这个数组[4,5,6,7,0,1,2] 你第一次切开,不管从哪切,你都会得到一个有序的数组,和一个旋转后的数组,怎么判断哪一半是有序的呢?很显然,如果第一个数,小于你切到前面的数,那么,这一半就是有序的,那另一半自然就是旋转后的数组。
public int search(int[] nums, int target) {
if(nums.length == 0) return -1;
if(nums.length == 1) {
if(nums[0] == target)
return 0;
return -1;
}
int l = 0;//代表low
int m= 0; //代表middle
int h = nums.length - 1; //high
while(l <= h) {
m =(h + l)/2; //每次都取中间的数
if(nums[m] == target) return m; //如果相等那么就返回
if(nums[l] <= nums[m]) {//当前一段为有序数组时
if(nums[l] <= target && target < nums[m]){ //并且在这个区间内,我们就缩小到这个区间
h = m - 1;
} else {
l= m + 1;//没有,那肯定是在那个区间
}
} else { //前一段是旋转数组,我们就去后一段找,后一段肯定是有序数组
if(nums[m] < target && target <= nums[h]){//同上
l = m +1;
} else {
h = m - 1;
}
}
}
return -1;
}