我的leetcode代码都已经上传到我的git
题干
假设按照升序排序的数组在预先未知的某个点上进行了旋转。
( 例如,数组 [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
想法
我们首先得认清这个事实,旋转以后nums的大小是先变大
然后跳跃间断点一般下降再上升
但是前半段的大小都大于后半段
于是比nums[0]大于等于的数都在前半段(注意等于也是嗷
比它小的都在后半段
于是我们就可以划分这俩段,对于每段都是递增的,就可以找到了。
第一种方法:分开查找,不再赘述
第二种:如果target在前半段,那么把后半段都设置成Integer.MAX_VALUE
那么整体都递增了,使用二分法就好
如果target在后半段,那么把前半段都设置成Integer.MIN_VALUE,
那么整体递减,二分法同理。
直接看代码吧
Java代码
package daily;
public class Search {
public int search(int[] nums, int target) {
//low high双指针用于二分法
int low=0;
int high=nums.length-1;
while(low<=high){
int mid=low+(high-low)/2;
//找到条件
if(nums[mid]==target){
return mid;
}
//target在前半段,注意等于
if(target>=nums[0]){
//mid在后半段,全设为正无穷
if(nums[mid]<nums[0]){
nums[mid]=Integer.MAX_VALUE;
}
}
//target在后半段
else {
//mid在前半段,全设为负无穷,注意等于
if(nums[mid]>=nums[0]){
nums[mid]=Integer.MIN_VALUE;
}
}
//二分法
if(nums[mid]<target){
low=mid+1;
}
else {
high=mid-1;
}
}
return -1;
}
public static void main(String[] args){
Search search=new Search();
int [] nums={4,5,6,7,0,1,2};
int [] n={1,3};
int target1=0 ,target2=3;
System.out.println(search.search(nums,target1));
System.out.println(search.search(nums,target2));
System.out.println(search.search(n,3));
}
}