我爱学算法之—— 二分查找(中)
一、搜索插入位置
题目解析
这道题,给定一个数组
nums
和一个目标值target
,让我们在数组nums
中找到目标值;如果目标值存在就返回它的下标,如果不存在就返回数target
被顺序插入的位置下标。
算法思路
这道题,我们可以使用暴力查找,时间复杂度是O(n)
。
从左到右遍历数组,找到值大于等于
target
的起始位置(如果target
不存在,那要找的就是target
顺序插入的位置)
题目要求我们要使用时间复杂度为O(log n)
的算法来解决,暴力解法的时间复杂度为O(n)
(暴力解法虽然可以通过这道题)
思考以下:数组nums
是有序的,且我们要找的是>=target
的起始位置,那也就是大于等于target
区间的左端点;
我们再随机挑选一个位置i
,区间[0,i-1]
中的数都是小于i
位置的数;区间[i+1 , n]
中的数都是大于i
位置的数的。
那我们就可以使用二分查找
首先定义
left
和right
指向数组的起始位置和结束位置。取
mid = left + (right - left)/2
比较
mid
位置的值和target
如果
nums[mid] > target
,就去左边区间查找;right = mid - 1
。如果
nums[mid] < target
,就去右边区间查找;left = mid + 1
。如果
nums[mid] == target
,就查找到了target
,就返回当前位置下标即可。这里因为我们要查找的是
>=target
区间的左端点,所以在遍历结束后,l
指向的就是target
顺序排序的插入位置。
代码实现
class Solution {
public:int searchInsert(vector<int>& nums, int target) {int l = 0, r = nums.size()-1;while(l <= r){int mid = (l + r)/2;if(nums[mid] < target)l = mid + 1;else if(nums[mid] > target)r = mid - 1;elsereturn mid;}return l;}
};
二、x 的平方根
题目解析
这道题,给定一个非负整数
x
,然我们计算它的算数平方根。注意:我们返回类型是一个整数,结果只保留整数部分,小数部分要舍去(也就是向下取整)
算法思路
对于这道题,暴力解法:
我们从
0
开始向后遍历,依次判断i
是不是我们要找的x
是算术平方根;这里可以从
x
开始向后遍历,查找到第一个i
的平方小于等于x
的位置。
暴力力解法遍历的区间是[1 , x],我们遍历过程中要判断i的平方是否等于x(或者大于)
但是我们思考一下[1 , x]
区间内数的平方它是递增的;
当我们遍历一个位置i
时,那[1, i-1]
区间内任意一个数的平方都小于i
的平方;区间[i+1 , x]
内的任意一个数的平方都是大于i
的平方的。
所以我们就可以使用二分查找来查找平方<=x
区间的左端点。
首先定义
left,right
(初始值为0
和x
),然后取mid = left +(right - left + 1)/2
- 如果
mid*mid
<x
:此时左边区间数的平方都是小于x
,肯定不会是最终结果;left = mid
。- 如果
mid*mid
>x
:此时右边区间数的平方都是大于x
,mid
平方也是大于x
的,肯定也不会是最终结果;right = mid - 1
。- 如果
mid*mid
==x
:此时就找到了x
的算数平方根,返回结果即可。最后遍历结束
left
和right
指向位置就是最终结果。
暴力解法遍历的区间是[1 , x]
,我们遍历过程中要判断i
的平方是否等于x
(或者大于)
代码实现
class Solution {
public:int mySqrt(int x) {long long left = 0,right = x;while(left < right){long long mid = left + (right - left + 1)/2;long long sum = mid * mid;if(sum > x)right = mid - 1;else if(sum < x)left = mid;elsereturn mid;}return left;}
};
三、山脉数组的峰顶索引
题目解析
对于这道题,给定一个数组
nums
,数组中的数据大小是山峰形状的(也就是先递增,然后再递减)现在我们要找到,山峰峰顶的索引值;也就是先递增再下降的转折点。
算法思路
暴力解法:
遍历数组,找到呈下降趋势的起始位置。
简单来说就是遍历数组,如果当前位置是小于下一个位置的值,也就是下降趋势就继续遍历;
如果当前位置是大于下一个位置的值的,也就是上升趋势就返回结果即可。
题目中说,数组中的值是先递增到一个值再递减的,那也就是说我们遍历一个位置i
时,只存在以下两种情况:
- 递增:
nums[i] > nums[i+1]
- 递减:
nums[i] < nums[i+1]
所以我们遍历一个位置
i
时:
- 如果当前是递增趋势(
nums[i] < nums[i+1]
),那区间[l,i]
就是递减的,我们要找的最终结果肯定是在区间[i+1, r]
中的;- 如果当前是递减趋势,也就是
nums[i] > nums[i+1]
,那区间[i+1, r]
就是递减的,而i
位置也可能是我们最终要找的结果,所以我们要找的最终结果肯定是在区间[l,i]
中的。所以这道题我们就可以使用二分查找来搜索山峰的峰顶位置。
二分查找
这里我们通过上面分析我们可以发现,我们可以将区间分成两部分,一部分是不满足条件的,一部分可能满足条件的;
而这里我们要找的是递减区间的左端点。
首先定义
left,right
(初始值为0
和n-1
),然后取mid = left +(right - left)/2
- 如果
nums[mid] < nums[mid+1]
:此时区间[left , mid]
是不满足条件的,就要去区间[mid+1 , right]
中查找最终结果。- 如果
nums[mid] > nums[mid+1]
:此时区间[mid+1 , right]
是不满足条件的,就要去区间[left , mid]
中查找最终结果。最后遍历结束,
left
和right
指向的就是最终结果。
代码实现
class Solution {
public: int peakIndexInMountainArray(vector<int>& arr) {int l = 0, r = arr.size()-1;while(l < r){int mid = (l + r)/2;if(arr[mid] > arr[mid+1])r = mid;elsel = mid+1;}return l;}
};
到这里本篇文章内容就结束了,感谢各位大佬的支持