模块三——二分:704.二分查找

前言

本系列博客是逐渐深入的过程,建议从零开始学习的友友不要跳过一些中间的博客。

二分查找算法简介

特点

最恶心,细节最多,最容易写出死循环的算法。

学习中的侧重点

算法原理

二分查找算法一定是数组有序的情况?答案显然是否定的,只要具有二段性就能使用二分查找算法。

模板

PS:不要死记硬背,要理解性记忆。

  1. 朴素的二分模板(easy但有局限)
  2. 查找左边界的二分模板(万能,细节多)
  3. 查找右边界的二分模板(万能,细节多)
//朴素二分模板
while(left <= right){
	int mid = left + (right - left) / 2;//防止越界,等价于left + (right - left + 1) / 2;
	//PS:为偶数时前者为中间两数的左数,后者为右数
	if(...)
		left = mid + 1;
	else if(...)
		right = mid - 1;
	else
		return ...;
}
//查找区间左端点的模板
while(left < right){
	int mid = left + (right - left) / 2;
	if(...)
		left = mid + 1;
	else right = mid;
}
//查找区间右端点的模板
while(left < right){
	int mid = left + (right - left + 1) / 2;
	if(...)
		left = mid;
	else right = mid - 1;

PS:分类讨论的代码,就题论题即可。

题目描述

题目链接:704.二分查找
在这里插入图片描述
题目很简单,就是在升序数组中搜索目标值target。

算法原理

解法一:暴力解法

直接遍历一遍数组即可,时间复杂度为O(N)

解法二:二分查找算法

算法流程

  • 定义 left ,right 指针,分别指向数组的左右区间。
  • 找到待查找区间的中间点 mid ,找到之后分三种情况讨论:
  1. arr[mid] == target 说明正好找到,返回 mid 的值;
  2. arr[mid] > target 说明 [mid,right] 这段区间都是⼤于 target 的,因此舍去右边区间,在左边[left, mid -1] 的区间继续查找,即让right = mid - 1 ,然后重复找mid过程;
  3. arr[mid] < target 说明 [left, mid]这段区间的值都是⼩于 target 的,因此舍去左边区间,在右边[mid + 1, right] 区间继续查找,即让 left =mid + 1 ,然后重复 找mid过程;
  • 当 left 与 right 错开时,说明整个区间都没有这个数,返回 -1 。

细节问题

循环结束的条件

left > right

为什么是正确的?

暴力解法是一次一次比较,但二分是利用了数组有序的特性,即二段性来通过比较一次来去掉多余的比较,所以暴力解法既然是正确的,那么二分也是正确的。

时间复杂度

第一次循环把区间划分成n/2,第二次是n/4,直到最坏情况下循环x次,而最后只剩下一个元素。n / 21,n / 22,n / 23…n / 2x——>2x = n——>x = logN。

代码实现

class Solution {
public:
    int search(vector<int>& nums, int target) {
        //二段性可用二分
        int left = 0;
        int right = nums.size() - 1;
        //结束条件
        while(left <= right){
            //直接加有可能会超出int的范围
            //int mid = (left + right) / 2;
            //换成减法防止溢出
            int mid = left + (right - left) / 2;
            if(nums[mid] > target){
                right = mid - 1;
            }
            else if(nums[mid] < target){
                left = mid + 1;
            }
            else{
                return mid;
            }
        }
        return -1;
    }
};
  • 25
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

全天

加油,大佬们!!!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值