二分查找算法专题(1)
找往期文章包括但不限于本期文章中不懂的知识点:
个人主页:我要学编程(ಥ_ಥ)-CSDN博客
所属专栏: 优选算法专题
目录
二分查找算法的介绍
704. 二分查找
34. 在排序数组中查找元素的第一个和 最后一个位置
35. 搜索插入位置
69. x的平方根
总结
二分查找算法的介绍
想必大家对这个算法应该不算陌生了,在C语言阶段就已经学习过了。 其是在暴力枚举的基础上进行优化的。例如:在一个有序数组中查找某个元素是否存在。

但是二分查找算法也有缺点,就是需要数据有二段性,不一定是数组全部有序。
二分查找算法其实也是双指针算法中对撞指针的一种拓展,主要是利用了数据的二段性。
下面我们就来进行练习。
704. 二分查找
题目:
给定一个
n个元素有序的(升序)整型数组nums和一个目标值target,写一个函数搜索nums中的target,如果目标值存在返回下标,否则返回-1。
示例 1:输入:nums = [-1,0,3,5,9,12], target = 9输出: 4 解释: 9 出现在 nums 中并且下标为 4示例 2:
输入:nums = [-1,0,3,5,9,12], target = 2输出: -1 解释: 2 不存在 nums 中因此返回 -1提示:
- 你可以假设
nums中的所有元素是不重复的。n将在[1, 10000]之间。nums的每个元素都将在[-9999, 9999]之间。
思路:这里既可以使用最简单的暴力枚举,也可以使用二分查找来解决。
代码实现:
暴力枚举:
class Solution {public int search(int[] nums, int target) {for (int i = 0; i < nums.length; i++) {if (nums[i] == target) {return i;}}return -1;}
}
二分查找:
class Solution {public int search(int[] nums, int target) {int left = 0;int right = nums.length-1;while (left <= right) { // 这里得判断=的情况int mid = (left+right) / 2; // 这里可能会有溢出的风险if (nums[mid] > target) {right = mid-1;} else if (nums[mid] < target) {left = mid+1;} else {return mid;}}return -1;}
}
注意:由于本题数据量不是很大,因此 mid = (left+right) / 2; 就不会溢出,但是当数据量非常大时,两者相加就会导致溢出。有小伙伴可能会有疑惑:left 为 0,right 在 int 中,为什么会导致溢出呢?确实这种情况是正常的,但是当第二次计算mid 且left 为上一次的mid 值呢?这就会溢出了。解决办法为:mid = left + (right - left)/2;上面这个题目只是来练练手,下面才开始真正的算法题。
34. 在排序数组中查找元素的第一个和 最后一个位置
题目:
给你一个按照非递减顺序排列的整数数组
nums,和一个目标值target。请你找出给定目标值在数组中的开始位置和结束位置。如果数组中不存在目标值
target,返回[-1, -1]。你必须设计并实现时间复杂度为
O(log n)的算法解决此问题。示例 1:
输入:nums = [5,7,7,8,8,10], target = 8 输出:[3,4]示例 2:
输入:nums = [5,7,7,8,8,10], target = 6 输出:[-1,-1]示例 3:
输入:nums = [], target = 0 输出:[-1,-1]提示:
0 <= nums.length <= 105-109 <= nums[i] <= 109nums是一个非递减数组-109 <= target <= 109
思路:题目说给的数组是非递减的,什么意思呢?

这里要查找的不是一个元素,而是一组连续的数据,也就是一段连续的子区间。 这里可能有小伙伴会想到我们前面学习的滑动窗口算法求子序列的问题。 但是这里不应该优先使用这个方法,因为滑动窗口算法是同向双指针, 而这里我们推测出了数据的特性,应该优先使用二分查找。
这里是要查找一组数据的端点下标,那么我们就可以直接忽略这组数据的中间,直接找端点即可。那么这就从查找一段数据,变为了查找两个值。但是新的问题又来了,怎么找端点呢?相信有聪明的小伙伴已经想到怎么做了。直接暴力枚举去遍历数组就完了。没错,这虽然是一个笨办法,但是总好过没有办法。
遍历的方式:从数组最左端开始遍历,找左端点,接着从数组最右端开始,找右端点即可。
代码实现:
class Solution {public int[] searchRange(int[] nums, int target) {int[] ans = {-1,-1};if (nums.length == 0) { // 排除特殊情况return ans;}// 找左端点int left = 0;while (left < nums.length && nums[left] != target) { // 防止越界left++;}if (left == nums.length) { // 数组中没有目标值return ans;} else {ans[0] = left;}// 找右端点int right = nums.length-1;while (right >= 0 && nums[right] != target) { // 防止越界right--;}if (right >= 0) {ans[1] = right;}return ans;}
}
虽然这是暴力枚举,但是从力扣上面的结果来看,还是不错的。

上面的方法可以说是流氓做法了,不符合题目的要求:用二分查找来解决。
二分查找同样还是去找符合数据的左端点和右端点。
寻找左端点过程:

寻找右端点过程(精简版):

上面处理这么多,其实就是在证明三件事:
1、根据查找的端点位置,从而划分了合法区域和非法区域,因为端点位置肯定是在有效区域内的。再根据 left 和 right 的相对位置来判断下一步的走向。
左端点:left = mid + 1 ---> 跳出非法区域;right = mid ---> 保留在合法区域。
右端点:left = mid ---> 保留在合法区域;right = mid -1 ---> 跳出非法区域。
2、在查找的过程中,中点的选取。根据查找的端点位置和第一点的结论,从而决定中点的位置。
左端点:right = mid 的特性可能会导致最后死循环,因此中点尽量要靠左,即 mid = left + (right-left) / 2。
右端点:left = mid 的特性可能会导致最后死循环,因此中点尽量要靠右,即 mid = left + (right-left +1) / 2。
3、 查找左端点和右端点的过程中,循环条件只能是 left < right,绝不能出现等于的情况,可能会导致死循环。因为一旦相遇并且结果满足 right 或者 left 不动的情况,那么就会死循环。
上面这些细节问题处理完之后,代码就比较好写了。
代码实现:
class Solution {public int[] searchRange(int[] nums, int target) {int[] ans = {-1,-1};if (nums.length == 0) { // 排除特殊情况return ans;}// 找左端点int left = 0;int right = nums.length-1;while (left < right) {int mid = left + (right-left) / 2; // 找靠左的位置if (nums[mid] >= target) {right = mid; // 保证在合法区域内} else {left = mid+1; // 保证有可能跳出非法区域}}// 走到这里,说明left与right相遇了if (nums[left] == target) { // 判断是否为左端点ans[0] = left; // left 与 right 都是可以的} else { // 说明数组中没有要找的数据return ans;}// 找右端点left = 0;right = nums.length-1;while (left < right) {int mid = left + (right-left+1) / 2; // 找靠右的位置if (nums[mid] <= target) {left = mid; // 保证在合法区域内} else {right = mid-1; // 保证有可能跳出非法区域}}// 走到这里,说明left与right相遇了if (nums[right] == target) { // 判断是否为右端点ans[1] = right; // left 与 right 都是可以的}return ans;}
}
还有两个要注意的地方:

因此数组中一旦存在我们要查找的数据的话,肯定是存在左右端点的。
35. 搜索插入位置
题目:
给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。
请必须使用时间复杂度为
O(log n)的算法。示例 1:
输入: nums = [1,3,5,6], target = 5 输出: 2示例 2:
输入: nums = [1,3,5,6], target = 2 输出: 1示例 3:
输入: nums = [1,3,5,6], target = 7 输出: 4提示:
1 <= nums.length <= 104-104 <= nums[i] <= 104nums为 无重复元素 的 升序 排列数组-104 <= target <= 104
思路:这里和第一题有点类似,但不同的是这一题的数组中可能不存在 target 这个数据。但是方法还是类似的。

当 [target,right] 区间是合法区间时,right = mid ---> 保证 right 在合法区间内,left = mid+1 ---> 保证 left 有可能进入合法区间,mid = left + (right - left) / 2 ---> 靠左的位置。同理,当[left,target]为合法区间时,也是类似的,这里就不过多赘述了。
代码实现:
1、当 [left, target] 是合法区间时:
class Solution {public int searchInsert(int[] nums, int target) {int left = 0;int right = nums.length-1;// 假设[left, target]是合法区间while (left < right) {int mid = left + (right-left+1) / 2;if (nums[mid] > target) {right = mid-1;} else {left = mid;}}// 判断是否存在if (nums[left] == target) { // 实际存在return left;} else { // 不存在// 判断是插入左边还是右边位置if (nums[left] > target) {return left;} else {return left+1;}}}
}
2、 当 [target,right] 是合法区间时:
class Solution {public int searchInsert(int[] nums, int target) {int left = 0;int right = nums.length-1;// 假设[target, right]是合法区间while (left < right) {int mid = left + (right-left) / 2;if (nums[mid] >= target) {right = mid;} else {left = mid+1;}}// 判断是否存在if (nums[left] == target) { // 实际存在return left;} else { // 不存在// 判断是插入左边还是右边位置if (nums[left] > target) {return left;} else {return left+1;}}}
}
69. x的平方根
题目:
给你一个非负整数
x,计算并返回x的 算术平方根 。由于返回类型是整数,结果只保留 整数部分 ,小数部分将被 舍去 。
注意:不允许使用任何内置指数函数和算符,例如
pow(x, 0.5)或者x ** 0.5。示例 1:
输入:x = 4 输出:2示例 2:
输入:x = 8 输出:2 解释:8 的算术平方根是 2.82842..., 由于返回类型是整数,小数部分将被舍去。提示:
0 <= x <= 231 - 1
思路: 题目让我们求一个大于等于0整数的算术平方根,并且对最终结果进行向下取整。
方法一:直接暴力枚举即可。
代码实现:
class Solution {// 暴力枚举public int mySqrt(int x) {if (x == 0 || x == 1) { // 排除特殊情况return x;}for (long i = 1; i <= x; i++) {if (i * i == x) {return (int)i;} else if (i * i > x) {return (int)i-1;}}return -1; // 这里只是过审}
}
注意:由于最后面的 return -1;只是为了让我们的代码编译通过,并不起实际的作用。
我们前面的暴力枚举就是把 [1,x] 之间的数据按照升序的方式挨个使了个遍。 从这里我们就可以使用二分查找算法了。

其实我们最终的目的就是为了找到大于或者的结果,然后再让大于的-1,等于的不变,而这些只能让 target 和 left 在一起。
代码实现:
class Solution {// 二分查找public int mySqrt(int x) {if (x == 0 || x == 1) { // 排除特殊情况return x;}long left = 1;long right = x;// 最终的结果是向下取整的,即 <= 是合法区域的while (left < right) {long mid = left + (right-left+1) / 2;if (mid*mid > x) {right = mid-1;} else {left = mid;}}// 找到了return (int)left;}
}
注意:
1、数据量是比较大的,因此相乘的结果会溢出,我们得用 long类型来接收。
2、这里的二分查找是不能使用第一道题的那种的。
其实没弄明白也没关系,这里反正就两种情况,可以直接去套用,再不济暴力枚举总可以了吧。
总结
1、对于查找固定的数据的情况,可以使用第一题中的二分查找方法:根据要查找的结果,进行比较分为三种情况——大于、等于、小于。
2、对于范围(区间)查找和不确定性查找的情况,可以使用我们后面画图推出来的二分查找:根据查找的结果,进行比较分为两种情况——合法区域、非法区域(根据要查找的数据进行分区),然后再分别更新 left 和 right——合法的一定要确保依旧存在于合法区域,非法的要确保有希望调到合法区域。再就是计算中点的方式和循环条件的确定,都是由 left 和 right 的变化来决定的(具体可见图)。
我们以后常用的也是第二种二分查找的方法。
好啦!本期 二分查找算法专题(1)的学习之旅就到此结束啦!我们下一期再一起学习吧!
相关文章:
二分查找算法专题(1)
找往期文章包括但不限于本期文章中不懂的知识点: 个人主页:我要学编程(ಥ_ಥ)-CSDN博客 所属专栏: 优选算法专题 目录 二分查找算法的介绍 704. 二分查找 34. 在排序数组中查找元素的第一个和 最后一个位置 35. 搜索插入位置 69. x的平…...
ACP科普:SoS不是救命
Scrum of Scrums(SoS)是一种用于协调多个Scrum团队之间工作的扩展框架,特别适用于大型项目或组织中有多个团队同时进行开发的情况。它帮助团队在保持敏捷性的同时,解决跨团队的依赖和协调问题。以下是对Scrum of Scrums的详细介绍…...
C++:模拟实现vector
目录 成员变量与迭代器 size capacity empty 迭代器有关函数 实现默认成员函数的前置准备 reserve 编辑 编辑 push_back 构造函数 无参构造 迭代器区间构造 n个val来进行构造 析构函数 拷贝构造函数 赋值重载 增删查改 clear resize pop_back inser…...
Leecode SQL 184. Department Highest Salary 找出tie
Department Highest Salary 注意!要找出 tie 的 highest salary! Write a solution to find employees who have the highest salary in each of the departments. Return the result table in any order. The result format is in the following ex…...
[Redis][典型运用][缓存]详细讲解
目录 0.什么是缓存?1.使用Redis作为缓存1.为什么用?2.如何用? 2.缓存的更新策略0.前言1.定期生成2.实时生成 3.缓存相关问题1.缓存预热(Cache Preheating)2.缓存穿透(Cache Penetration)3.缓存雪崩(Cache Avalanche)4.缓存击穿(Cache Breakdo…...
GPG error golang 1.19
1. 问题描述及原因分析 在飞腾2000的服务器,OS为Kylin Linux Advanced Server release V10环境下,docker版本为18.09.0(docker-engine-18.09.0-101.ky10.aarch64),基于容器镜像golang:1.19编译新的容器镜像࿰…...
Linux如何查看每个文件及文件夹的大小
查看当前目录下每个文件夹的大小,包括其内部所有文件: du -sh *-s:仅显示每个文件夹的总大小,而不是每个文件。-h:以人类可读的格式显示。...
Word样式的同步与重置
有时候我们需要修改Word中的样式,实现排版的个性化。 如何同步样式到其他电脑上? Word中的样式是由Normal.dotm文件控制的,对样式所有的设置和修改,都会保存到这个问题件中,所以我们只需要在设置好样式以后ÿ…...
力扣 —— 跳跃游戏
题目一(中等) 给你一个非负整数数组 nums ,你最初位于数组的 第一个下标 。数组中的每个元素代表你在该位置可以跳跃的最大长度。 判断你是否能够到达最后一个下标,如果可以,返回 true ;否则,返回 false 。 示例 1&…...
SOCKS5代理和HTTP代理哪个快?深度解析两者的速度差异
在现代互联网环境中,使用代理IP已经成为了许多人日常生活和工作的必备工具。无论是为了保护隐私,还是为了访问某些特定资源,代理IP都扮演着重要的角色。今天,我们就来聊聊SOCKS5代理和HTTP代理,看看这两者到底哪个更快…...
工具介绍---效率高+实用
Visual Studio Code (VS Code) 功能特点: 智能代码提示:内置的智能代码提示功能可以自动完成函数、变量等的输入,提高代码编写速度。插件丰富:支持成千上万的扩展插件,例如代码片段、主题、Linting等,能够…...
本地部署开源在线PPT制作与演示应用PPTist并实现异地远程使用
文章目录 前言1. 本地安装PPTist2. PPTist 使用介绍3. 安装Cpolar内网穿透4. 配置公网地址5. 配置固定公网地址 前言 本文主要介绍如何在Windows系统环境本地部署开源在线演示文稿应用PPTist,并结合cpolar内网穿透工具实现随时随地远程访问与使用该项目。 PPTist …...
leetcode_238:除自身以外数组的乘积
给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂…...
网络协议详解--IPv6
IPv6产生背景 (1)地址空间的耗尽:因特网呈指数级发展,导致IPv4地址空间几乎耗尽。虽然采用了子网划分、CIDR和NAT地址转换技术,但这没有从根源解决地址耗尽的问题 (2)IP层安全需求的增长&#x…...
阿里云域名注册购买和备案
文章目录 1、阿里云首页搜索 域名注册2、点击 控制台3、域名控制台 1、阿里云首页搜索 域名注册 2、点击 控制台 3、域名控制台...
【经典机器学习算法】谱聚类算法及其实现(python)
🌈 个人主页:十二月的猫-CSDN博客 🔥 系列专栏: 🏀深度学习_十二月的猫的博客-CSDN博客 💪🏻 十二月的寒冬阻挡不了春天的脚步,十二点的黑夜遮蔽不住黎明的曙光 目录 1. 前言 2. 前…...
【Linux】Linux环境基础开发工具使用
Linux开发工具 Linux编辑器-vim使用 1. vim的基本概念 vim的三种模式,分别是命令模式(command mode)、插入模式(Insert mode)和底行模式(last line mode)。 正常/普通/命令模式: …...
Halcon基础系列1-基础算子
1 窗口介绍 打开Halcon 的主界面主要有图形窗口、算子窗口、变量窗口和程序窗口,可拖动调整位置,关闭后可在窗口下拉选项中找到。 2 显示操作 关闭-dev_close_window() 打开-dev_open_window (0, 0, 712, 512, black, WindowHandle) 显示-dev_display(…...
【AI大模型】深入Transformer架构:编码器部分的实现与解析(上)
目录 🍔 编码器介绍 🍔 掩码张量 2.1 掩码张量介绍 2.2 掩码张量的作用 2.3 生成掩码张量的代码分析 2.4 掩码张量的可视化 2.5 掩码张量总结 🍔 注意力机制 3.1 注意力计算规则的代码分析 3.2 带有mask的输入参数: 3.…...
spring学习日记-day7-整合mybatis
一、学习目标 spring整合MyBatis的原理主要涉及到将MyBatis的Mapper映射文件交由Spring容器管理,并将其注入到MyBatis的SqlSessionFactory中,从而实现两者的整合。 二、整合mybatis 1.写一个mybatis测试案例 项目结构: 1.数据库 CREATE DA…...
python打卡day49
知识点回顾: 通道注意力模块复习空间注意力模块CBAM的定义 作业:尝试对今天的模型检查参数数目,并用tensorboard查看训练过程 import torch import torch.nn as nn# 定义通道注意力 class ChannelAttention(nn.Module):def __init__(self,…...
前端倒计时误差!
提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...
【论文笔记】若干矿井粉尘检测算法概述
总的来说,传统机器学习、传统机器学习与深度学习的结合、LSTM等算法所需要的数据集来源于矿井传感器测量的粉尘浓度,通过建立回归模型来预测未来矿井的粉尘浓度。传统机器学习算法性能易受数据中极端值的影响。YOLO等计算机视觉算法所需要的数据集来源于…...
UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)
UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中,UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化…...
【JavaWeb】Docker项目部署
引言 之前学习了Linux操作系统的常见命令,在Linux上安装软件,以及如何在Linux上部署一个单体项目,大多数同学都会有相同的感受,那就是麻烦。 核心体现在三点: 命令太多了,记不住 软件安装包名字复杂&…...
大数据学习(132)-HIve数据分析
🍋🍋大数据学习🍋🍋 🔥系列专栏: 👑哲学语录: 用力所能及,改变世界。 💖如果觉得博主的文章还不错的话,请点赞👍收藏⭐️留言Ǵ…...
.Net Framework 4/C# 关键字(非常用,持续更新...)
一、is 关键字 is 关键字用于检查对象是否于给定类型兼容,如果兼容将返回 true,如果不兼容则返回 false,在进行类型转换前,可以先使用 is 关键字判断对象是否与指定类型兼容,如果兼容才进行转换,这样的转换是安全的。 例如有:首先创建一个字符串对象,然后将字符串对象隐…...
Angular微前端架构:Module Federation + ngx-build-plus (Webpack)
以下是一个完整的 Angular 微前端示例,其中使用的是 Module Federation 和 npx-build-plus 实现了主应用(Shell)与子应用(Remote)的集成。 🛠️ 项目结构 angular-mf/ ├── shell-app/ # 主应用&…...
LeetCode - 199. 二叉树的右视图
题目 199. 二叉树的右视图 - 力扣(LeetCode) 思路 右视图是指从树的右侧看,对于每一层,只能看到该层最右边的节点。实现思路是: 使用深度优先搜索(DFS)按照"根-右-左"的顺序遍历树记录每个节点的深度对于…...
LangChain知识库管理后端接口:数据库操作详解—— 构建本地知识库系统的基础《二》
这段 Python 代码是一个完整的 知识库数据库操作模块,用于对本地知识库系统中的知识库进行增删改查(CRUD)操作。它基于 SQLAlchemy ORM 框架 和一个自定义的装饰器 with_session 实现数据库会话管理。 📘 一、整体功能概述 该模块…...
