二分查找法思路
直接求解问题
题目找出一个整齐排列的升序数组中目标值target第一次出现和最后一次出现的位置。
实例:[2,2,3,3,4,5,5] target=4 return [4,4];
若没有目标值则返回[-1,-1]。
思路:根据题目要求找到两个位置。首先确定范围区间[left,right],然后通过mid判断创立条件。
查找第一次出现位置时
mid=(left+right)>>>1
1.nums[mid]>target mid及其右边范围排除,缩小范围区间为[left,mid-1]。
2.nums[mid]<target mid及其左边范围排除,缩小范围区间为[mid+1,right]。
3.nums[mid]==target mid右边范围排除,缩小范围区间为[left,mid];
查找最后一次出现位置时
mid=(left+right+1)>>>1,需要从高位查找,否则容易死循环。
1.nums[mid]>target mid及其右边范围排除,缩小范围区间为[left,mid-1]。
2.nums[mid]<target mid及其左边范围排除,缩小范围区间为[mid+1,right]。
3.nums[mid]==target mid左边范围排除,缩小范围区间为[mid,right]。
public int[] searRange(int[] nums,int target){
int len=nums.length;
if(len==0){
return new int[]{-1,-1};
}
int fastPosition = finFastPosition(nums,target);
if(fastPosition==-1){
return new int[]{-1,-1};
}
int lastPosition = finaLastposition(nums,target);
return new int[]{fastPosition,lastPosition};
}
private int finaLastposition(int[] nums, int target) {
int left=0;
int right=nums.length-1;
while(right>left){
int mid = (left+right+1)>>>1;
if(nums[mid]<target){
left=mid+1;
}
else if(nums[mid]>target){
right=mid-1;
}else {
left=mid;
}
}if(nums[right]==target){
return right;
}
return -1;
}
private int finFastPosition(int[] nums, int target) {
int left=0;
int right = nums.length-1;
while(right>left){
int mid = (left+right)>>>1;
if(nums[mid]>target){
right=mid-1;
}else if(nums[mid]<target){
left=mid+1;
}else{
right=mid;
}
}if(nums[left]==target){
return left;
}
return -1;
}