搜索旋转排序数组java实现

力扣 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;
	  }

你可能感兴趣的:(算法学习,JAVA学习)