优选算法——二分查找


👇作者其它专栏

《数据结构与算法》《算法》《C++起始之路》


二分查找相关题解

1.二分查找

算法思路:

a.定义left,right指针,分别指向数组的左右区间。

b.找到待查找区间的中间点mid,找到后分三种情况讨论:

        i.arr[mid]==target说明正好找到,返回mid的值;

        ii.arr[mid]>target说明[mid,right]这段区间都是大于target的,因此舍去右边区间,在左边[left,mid-1]的区间继续查找,即让right=mid-1,然后重复b过程;

        iii.arr[mid]<target说明[left,mid]这段区间的值都是小于target的,因此舍去左边区间,在右边区间[mid+1,right]区间继续查找,即让left=mid+1,然后重复b过程;

c.当left与right错开时,说明整个区间都没有这个数,返回-1。

//法一:遍历 //法二:二分查找 class Solution { public: int search(vector<int>& nums, int target) { int left=0,right=nums.size()-1; while(left<=right){ //int mid=(left+right)/2;可能会溢出 int mid=left+(right-left)/2; if(nums[mid]<target) left=mid+1; else if(nums[mid]>target) right=mid-1; else return mid; } return -1; } };

2.在排序数组中查找元素的第一个和最后一个位置

算法思路:

用的还是二分思想,就是根据数据的性质,在某种判断条件下将区间一分为二,然后舍去其中一个区间,然后在另一个区间内查找。

以下用x表示该元素,resLeft表示左边界,resRight表示右边界。

寻找左边界思路:

●寻找左边界:

        ●可以注意到一左边界划分的的两个区间的特点:

                ▢左边区间[left,resLeft-1]都是小于x的;

                ▢右边区间(包括左边界)[resLeft,right]都是大于等于x的;

●因此,关于mid的落点,我们可以分为以下两种情况:

       ●当我们的mid落在[left,resLeft-1]区间时,即arr[mid]<target。说明[left,mid]都是可以舍去的,此时更新left到mid+1的位置,继续在[mid+1,right]上寻找左边界;

       ●当mid落在[resLeft,right]的区间的时候,也就是arr[mid]>=target。说明[mid+1,right](因为mid可能是最终结果,不能舍去)是可以舍去的,此时更新right到mid的位置,继续在[left,mid]上寻找左边界;

●由此,就可以通过二分,来快速寻找左边界;

 注意:这里找中间元素需要向下取整。

因为后续移动左右指针时:

●左指针:left=mid+1,是会向后移动的,因此区间数会缩小的;

●右指针:right=mid,可能会原地踏步(如:若向上取整,如果剩下两个元素,left==1,right==2,mid==2。更新区间后,left,right,mid的值没有改变,就会陷入死循环)。

因此一定要注意,当right==mid时,要向下取整。

寻找右边界思路:

●寻找右边界:

        ●用resRight表示右边界;

        ●我们注意到右边界的特点:

                ▢左边区间(包括右边界)[left,resRight]都是小于等于x的;

                ▢右边区间[resRight+1,right]都是大于x的;

●因此,关于mid的落点,我们可以分为下面两种情况:

        ●当我们的mid落在[left,resRight]区间时,说明[left,mid-1](mid不可以舍去,因为可能是最终结果)都是可以舍去的,此时更新left到mid的位置;

        ●当mid落在[resRight+1,right]的区间的时候,说明[mid,resRight]内元素是可以舍去的,此时更新right到mid-1位置;

●由此,就可以通过二分,来快速寻找右边界;

注意:这里找中间元素需要向上取整。

因为后续移动左右指针的时候:

●左指针:left=mid,可能会原地踏步(如:若向下取整,如果剩下1,2两个元素,left==1,right==2,mdi==1。更新区间之后,left,right,mid的值没有改变,就会陷入死循环)。

右指针:right=mid-1,是会向前移动的,因此区间是会缩小的;

因此一定要注意,当right=mid-1时,要向上取整。

二分查找算法总结:

1.关于什么时候用三段式,还是二段式中的某一个,一定不要强行去用,而是通过具体的问题分析情况,根据查找区间的变化确定指针的转移过程,从而选择一个模板。

2.当选择两段式的模板时:

●在求mid时,只有right-1的情况下,才会向上取整(即+1,取中间数时)

class Solution { public: vector<int> searchRange(vector<int>& nums, int target) { //数组为空时 if(nums.size()==0) return {-1,-1}; int begin=0; int left=0,right=nums.size()-1; //求区间左端点 while(left<right){ int mid=left+(right-left)/2; if(nums[mid]<target) left=mid+1; else right=mid; } if(nums[left]!=target) return {-1,-1}; else begin=left; //求区间右端点 left=0,right=nums.size()-1; //left没必要重新置为0,因为它查找左端点后,一定不会超过右端点 while(left<right){ int mid=left+(right-left+1)/2; if(nums[mid]<=target) left=mid; else right=mid-1; } return {begin,right}; } };

3.搜索插入位置

算法思路:

a.分析插入位置左右两侧区间上元素的特点:

        设插入位置的坐标为index,根据插入位置的特点可以知道:

        ●[left,index-1]内的所有元素均是小于target的;

        ●[index,right]内的所有元素均是大于等于target的;

b.设left为本轮查询的左边界,right为本轮查询的右边界。根据mid位置元素的信息,分析下一轮查询的区间:

        ●当nums[mid]>=target时,说明mid落在了[index,right]区间上,mid左边包括mid本身,可能是最终结果,所以我们接下来查找的区间在[left,mid]上。因此,更新right到mid位置,继续查找。

        ●当nums[mid]<target时,说明mid落在了[left,index-1]区间上,mid右边但不包括mid本身,可能是最终结果,索引我们接下来查找的区间[mid+1,right]上。因此,更新left到mid+1位置,继续查找。

c.直到我们的查找区间长度变为1,即left==right时,left或right所在的位置就是我们要找的结果。

class Solution { public: int searchInsert(vector<int>& nums, int target) { int left=0,right=nums.size(); while(left<right){ int mid=left+(right-left)/2; if(nums[mid]<target) left=mid+1; else right=mid; } return left; } };

4.x 的平方根

算法思路一(暴力):

依次枚举【0,x】之间的所有数i:(这里没有必要研究是否枚举到x/2还是x/2+1。因为我们找到结果之后直接就返回了,往后的情况就不会再判断。反而研究枚举空间,既 耽误时间,又可能出错)

●若i*i==x,直接返回x;

●若i*i>x,说明之前的一个数是结果,返回i-1

由于i*i可能超过int的最大值,因此使用long long类型

class Solution{ public: int mySqrt(int x){ //防止溢出 long long i=0; for(i=0;i<=x;i++){ if(i*i==x) return i; if(i*i>x) return i-1; } return -1; } };

算法思路二(二分):

设x的平方根的最终结果为index:

a.分析index左右两次数据的特点:

        ●【0,index】之间的元素平方后都是小于等于x的;

        ●【index+1,x】之间的元素,平方后都是大于x的。

由此可以使用二分查找算法

//法一:循环遍历,平法大于x,即找到(此数-1) //法二:二分 class Solution { public: int mySqrt(int x) { //可将区间分为小于等于x的 大于x的 int left=1,right=x; if(x<1) return 0; while(left<right){ //long long 防止溢出 long long mid=left+(right-left+1)/2; if(mid*mid<=x) left=mid; else right=mid-1; } return left; } };

5.山脉数组的峰顶索引

算法思路一(暴力):

顶峰的特点:比两边的元素都要大。

因此,我们可以遍历数组内的每一个元素,找到某一个元素比两边的元素大即可。

class Solution { public: int peakIndexInMountainArray(vector<int>& arr) { int n=arr.size(); //遍历数组内每一个元素,直到找到峰顶 for(int i=1;i<n-1;i++){ //峰顶满足条件 if(arr[i]>arr[i-1]&&arr[i]>arr[i+1]) return i; } return -1; } };

算法思路二(二分):

1.分析峰顶位置的数据特点,以及山峰两旁的数据的特点:

●峰顶数据特点:arr[i]>arr[i-1]&&arr[i]>arr[i+1];

●峰顶左边的数据特点:arr[i]>arr[i-1]&&arr[i]<arr[i+1],即呈上升趋势;

●峰顶右边数据的特点:arr[i]<ar[i-1]&&arr[i]>arr[i+1],即呈下降趋势。

2.因此,根据mid位置的信息,我们可以分为下面三种情况:

●若mid位置呈上升趋势,说明我们接下来要在【mid+1,right】区间继续搜索;

●若mid位置呈下降趋势,说明我们接下来要在【left,mid-1】区间搜索;

●若mid位置就是山峰,直接返回结果。

class Solution { public: int peakIndexInMountainArray(vector<int>& arr) { //峰顶一定不会位于首尾 int left=1,right=arr.size()-2; while(left<right){ int mid=left+(right-left+1)/2; if(arr[mid]>arr[mid-1]) left=mid; else right=mid-1; } return left; } };

6.寻找峰值

解法思路(二分):
寻找二段性:

任取一点i,与下一个点i+1,会有如下两种情况:

●arr[i]>arr[i+1]:此时【左侧区域】一定会存在山峰(因为最左侧是负无穷),那么我们可以取左侧寻找结果;

●arr[i]<arr[i+1]:此时【右侧区域】一定会存在山峰(因为最右侧是负无穷),那么我们可以取右侧寻找结果。

当我们找到【二段性】时,就可以尝试用【二分查找】算法解决问题。

//法一:从前向后遍历,分情况讨论 //法二:二分 class Solution { public: //3种情况,1一直递减;2一直递增;3有增有减 int findPeakElement(vector<int>& nums) { int left=0,right=nums.size()-1; while(left<right){ int mid=left+(right-left)/2; //左边一定存在峰值,右边不一定[左边] [右边] if(nums[mid]<nums[mid+1]) left=mid+1; //右边一定存在峰值,左边不一定 else right=mid; } return left; } };

7.寻找旋转排序数组中的最小值

算法思路(二分):

c点就是我们要求的点。

二分的本质:找到一个判断标准,使得查找区间能够一分为二。

通过图像我们可以发现,【A,B】区间内的点都是严格大于D点的值的,C点的值是严格小于D的点的值的。但是当【C,D】区间只有一个元素的时候,C点的值是可能等于D点的值的。

因此,初始化左右两个指针left,right:

然后根据mid的落点,我们可以这样划分下一次查询的区间:

●当mid在【A,B】区间的时候,也就是mid位置的值严格大于D点的值,下一次查询区间在【mid+1,right】上;

●当mid在【C,D】区间的时候,也就是mid位置的值严格小于等于D点的值,下次查询区间在【left,mid】上。

当区间长度变成1的时候,就是我们要找的结果。

class Solution { public: int findMin(vector<int>& nums) { int left=0,right=nums.size()-1; int x=nums[right]; while(left<right){ int mid=left+(right-left)/2; //与数组中最后一个值比较 if(nums[mid]>x) left=mid+1; else right=mid; } return nums[left]; } };

也可以用左侧为基准值,但要注意排除数组为升序的情况:

class Solution { public: int findMin(vector<int>& nums) { int left=0,right=nums.size()-1; int x=nums[left];//以左端点为基准值 if(x<nums[right]) return nums[left]; while(left<right){ int mid=left+(right-left)/2; if(nums[mid]>=x) left=mid+1;//此时左端点一定不是最小值 else right=mid; } return nums[left]; } };

8.点名

算法思路(二分):

在这个升序的数组中,我们发现:

●在第一个缺失位置的左边,数组内的元素都是与数组的下标相等的;

●在第一个缺失位置的右边,数组内的元素与数组下标是不相等的。

因此,我们可以利用这个【二段性】,来使用【二分查找】算法。

//法一:直接遍历 法二:桶思想 法三:位运算(异或) 法四:数学公式(高斯) //法五:二分 class Solution { public: int takeAttendance(vector<int>& records) { int left=0,right=records.size()-1; while(left<right){ int mid=left+(right-left)/2; if(mid==records[mid]) left=mid+1; else right=mid; } //防止缺失的是最后一个数字 if(left==records[left]) return left+1; return left; } };

Read more

【大模型应用篇】用 OpenClaw + 飞书打造 7x24 小时服务器运维机器人

【大模型应用篇】用 OpenClaw + 飞书打造 7x24 小时服务器运维机器人

前言 本文基于OpenClaw,也是最近超火的可在本地运行的AI Agent网关,记录从零搭建通过飞书对话管理服务器运维机器人的全过程。该机器人支持随时随地通过飞书查看服务器状态、检索日志、管理进程,其核心机制在于:由OpenClaw将聊天平台(飞书等)的消息路由至大模型,模型调用本地工具(如Shell、文件系统、浏览器)执行相应任务,最终将结果自动返回至飞书会话中,实现自动化运维交互。 架构概览 飞书 App (WebSocket 长连接)         ↕ OpenClaw Gateway (服务器上 systemd 常驻)         ↕ AI 模型 (DeepSeek v3.2/GLM 4.7)         ↕ 服务器 Shell (受白名单限制的命令执行) 核心组件: * OpenClaw Gateway:Agent 网关,管理会话、工具调用、渠道连接 * 飞书插件:通过

By Ne0inhk
Clawdbot(Moltbot) 飞书机器人配置,体验老板和助手沟通的感觉

Clawdbot(Moltbot) 飞书机器人配置,体验老板和助手沟通的感觉

一、背景说明 Clawdbot可以24小时待命(参考配置方式:Clawdbot(Moltbot) windows安装配置教程(含各种问题处理)),但是网页端使用起来比毕竟没那么方便,然而clawdbot支持多种渠道交互,这也正是这个AI助理的魅力所在,想想飞书发送一个消息,一个任务就完成了,这不就是老板指挥我做事的方式吗,来赶紧体验一波老板的感觉~ 二、飞书机器人创建 飞书开放平台构建机器人:https://open.feishu.cn/ 记录App ID 和 App Secret,一会要用: 三、自动安装插件 项目地址:https://github.com/m1heng/Clawdbot-feishu 这时候,就可以发挥clawdbot的能力了,直接让clawdbot给我安装: 我要安装飞书机器人,帮我按照这个命令安装:Clawdbot plugins install @m1heng-clawd/feishu 到这个过程有点慢,安装了好一会没反应,我开始问了: 又过了好一会没反应,

By Ne0inhk

Mac Mini M4 跑 AI 模型全攻略:从 Ollama 到 Stable Diffusion 的保姆级配置指南

Mac Mini M4 本地AI模型实战:从零构建你的个人智能工作站 最近身边不少朋友都在讨论,能不能用一台小巧的Mac Mini M4,搭建一个属于自己的AI开发环境。毕竟,不是每个人都有预算去租用云端的高性能GPU,也不是所有项目都适合把数据传到云端处理。我折腾了大概两周,从Ollama到Stable Diffusion,把整个流程走了一遍,发现M4芯片的潜力远超预期。这篇文章,就是把我踩过的坑、验证过的有效配置,以及一些提升效率的小技巧,毫无保留地分享给你。无论你是想本地运行大语言模型进行对话和创作,还是想离线生成高质量的AI图像,这篇指南都能帮你把Mac Mini M4变成一个得力的AI伙伴。 1. 环境准备与基础配置 在开始安装任何AI工具之前,确保你的系统环境是干净且高效的,这能避免后续无数莫名其妙的依赖冲突。Mac Mini M4出厂预装的是较新的macOS版本,但这还不够。 首先,打开“系统设置” -> “通用” -> “软件更新”,确保你的macOS已经更新到可用的最新版本。苹果对Metal图形API和神经网络引擎的优化通常会随着系统更新而提升,这对于后续运

By Ne0inhk
知识库问答机器人:基于SpringAI+RAG的完整实现

知识库问答机器人:基于SpringAI+RAG的完整实现

一、引言 随着大语言模型的快速发展,RAG(Retrieval-Augmented Generation)技术已成为构建知识库问答系统的核心技术之一。本文将带领大家从零开始,使用Spring AI框架构建一个支持文档上传的知识库问答机器人,帮助大家深入理解RAG技术的核心原理和实践应用。 1.1 什么是RAG? RAG(检索增强生成)是一种结合了信息检索和文本生成的技术。它的基本工作流程是: 用户提出问题 系统从知识库中检索相关信息 大语言模型基于检索到的信息生成答案 从系统设计角度触发,RAG 的核心作用可以被描述为: 在LLM调用生成响应之前,由系统动态构造一个“最小且相关的知识上下文”。 请注意两个关键词: 动态 :每次问题都不同,检索的知识也不同(比如用户问 A 产品时找 A 的文档,问 B 产品时找 B 的文档) 最小 :只注入必要信息(比如用户问 “A 产品的定价”,就只塞定价相关的片段,而非整份产品手册) RAG可以有效的弥补上下文窗口的先天不足:不再需要把所有知识塞进窗口,

By Ne0inhk