二分查找看这篇就够了!Java 版超详细讲解+高频题解

news2026/3/16 12:51:20
二分查找看这篇就够了Java 版超详细讲解高频题解大家好今天我们来彻底吃透二分查找。作为算法面试、笔试中的“常青树”它是必考且基础的核心知识点看似只有“左右指针中间值对比”这一个简单逻辑但实际应用中边界处理、区间收缩、题型变形等细节极易出错很多人刷题时常常陷入死循环或边界遗漏的困境。今天我会用Java 语言从基础原理入手拆解标准模板再结合LeetCode高频真题实战演练一步步带你吃透二分查找轻松应对各类面试考点。一、什么是二分查找二分查找又称折半查找是一种高效的查找算法其核心适用前提有两个一是数组必须有序升序或降序均可本文默认以升序为例二是数组中无重复元素有重复元素的场景会在后续边界查找中专门讲解。核心思想通俗解读每次查找时将当前的搜索区间一分为二锁定中间位置的元素以此为基准进行判断通过中间元素与目标值的大小对比直接舍弃不可能包含目标值的一半区间无需逐一遍历仅在剩下的另一半区间中继续重复“分半-判断-舍弃”的流程直到找到目标值或确定目标值不存在时间复杂度为O(log n)n为数组长度相较于暴力查找的O(n)效率提升极为明显尤其在数据量较大如n10000时差距会格外突出。一句话总结有单调性就可以二分。这里的单调性不仅指数组整体有序只要能将搜索区间划分为“满足条件”和“不满足条件”两部分且两部分边界清晰就能用二分思想解题。二、最基础模板LeetCode 704 二分查找这是二分查找的入门经典题也是所有二分变形题的基础吃透这道题就能掌握二分查找的核心框架后续的边界查找、真题实战都能以此为基础延伸。题目给定一个 n 个元素的有序、无重复整型数组 nums 和一个目标值 target写一个函数搜索 nums 中的 target。如果目标值存在返回其下标如果不存在返回 -1。题目链接LeetCode 704. 二分查找Java 代码class Solution { public int search(int[] nums, int target) { int left 0; int right nums.length - 1; while (left right) { int mid left (right - left) / 2; // 防止溢出替代(leftright)/2 if (nums[mid] target) { return mid; // 找到目标值直接返回下标 } else if (nums[mid] target) { left mid 1; // 目标值在右侧区间缩小左边界 } else { right mid - 1; // 目标值在左侧区间缩小右边界 } } return -1; // 循环结束仍未找到返回-1 } }关键点避坑重点循环条件 left right这里必须包含“等于”因为当left和right重合时当前位置的元素还未被判断若漏掉“等于”会导致该位置的目标值被遗漏出现查找失败的错误。mid 不要写成 (leftright)/2当left和right均为较大的整数如接近int类型的最大值时两者相加会超出int类型的取值范围导致数值溢出进而出现计算错误而left (right - left)/2 等价于 (leftright)/2却能有效避免溢出问题。区间更新必须是mid1/mid-1否则会死循环因为当nums[mid]不等于target时该位置已被排除若仅更新为leftmid或rightmid会导致区间无法缩小陷入无限循环例如left1、right2mid1若nums[mid]targetleft仍设为1区间始终不变。三、二分最重要的能力找边界在实际面试中单纯查找目标值的基础题占比极低更多的是变形题——找左边界、右边界、插入位置这类题目核心考察对区间收缩的精准控制也是二分查找的难点所在掌握这部分就能应对80%的二分变形题。1. 找左边界第一个等于 target 的位置适用场景数组中存在重复元素需要找到目标值第一次出现的位置例如数组[1,2,2,2,3]target2左边界为1。private int leftBound(int[] nums, int target) { int left 0; int right nums.length - 1; while (left right) { int mid left (right - left) / 2; if (nums[mid] target) { left mid 1; // 目标在右侧左边界右移 } else { right mid; // 目标在左侧或当前位置右边界左移保留mid } } // 循环结束后leftright判断该位置是否为目标值 return nums[left] target ? left : -1; }2. 找右边界最后一个等于 target 的位置适用场景数组中存在重复元素需要找到目标值最后一次出现的位置例如数组[1,2,2,2,3]target2右边界为3。private int rightBound(int[] nums, int target) { int left 0; int right nums.length - 1; while (left right) { int mid left (right - left 1) / 2; // 向上取整避免死循环 if (nums[mid] target) { right mid - 1; // 目标在左侧右边界左移 } else { left mid; // 目标在右侧或当前位置左边界右移保留mid } } // 循环结束后leftright判断该位置是否为目标值 return nums[left] target ? left : -1; }记忆口诀快速掌握避免混淆找左边界mid 向下取整rightmid优先收缩右边界保留左侧可能的边界找右边界mid 向上取整leftmid优先收缩左边界保留右侧可能的边界循环条件一律用left right循环结束后left与right重合再判断是否为目标值避免边界遗漏。四、LeetCode 高频真题Java 版理论结合实战才是掌握二分的关键下面精选5道LeetCode高频真题覆盖边界查找、数值计算、旋转数组等常见场景每道题都附带详细解析和Java代码帮你巩固知识点应对面试实战。题目 1LeetCode 34 在排序数组中查找元素的第一个和最后一个位置这道题是左右边界查找的直接应用将前面讲解的左边界、右边界函数结合起来就能快速解题也是面试中最常考的边界类题目之一。class Solution { public int[] searchRange(int[] nums, int target) { // 先判断数组为空的特殊情况避免空指针异常 if (nums null || nums.length 0) return new int[]{-1, -1}; // 调用左边界、右边界函数获取两个边界下标 int left leftBound(nums, target); int right rightBound(nums, target); // 返回结果数组若目标值不存在两个边界均为-1 return new int[]{left, right}; } // 左边界查找函数复用前面的实现 private int leftBound(int[] nums, int target) { int left 0, right nums.length - 1; while (left right) { int mid left (right - left) / 2; if (nums[mid] target) left mid 1; else right mid; } return nums[left] target ? left : -1; } // 右边界查找函数复用前面的实现 private int rightBound(int[] nums, int target) { int left 0, right nums.length - 1; while (left right) { int mid left (right - left 1) / 2; if (nums[mid] target) right mid - 1; else left mid; } return nums[left] target ? left : -1; } }题目 2LeetCode 35 搜索插入位置这道题是左边界查找的变形核心需求是如果数组中存在target返回其下标如果不存在返回它应该被插入的位置保证插入后数组依然有序。解题关键是找到第一个大于等于target的位置即为插入位置。class Solution { public int searchInsert(int[] nums, int target) { int left 0; int right nums.length - 1; while (left right) { int mid left (right - left) / 2; if (nums[mid] target) { left mid 1; // 目标在右侧左边界右移 } else { right mid; // 目标在左侧或当前位置保留mid收缩右边界 } } // 最后判断若当前元素小于target插入到当前位置后面否则插入到当前位置 return nums[left] target ? left 1 : left; } }题目 3LeetCode 69 x 的平方根这道题是二分查找在数值计算中的应用需求是求x的算术平方根只保留整数部分例如x8平方根是2.828返回2。核心思路是找到最大的整数mid使得mid² ≤ x这个mid就是x的算术平方根的整数部分。class Solution { public int mySqrt(int x) { // 特殊情况处理x1时算术平方根小于1整数部分为0 if (x 1) return 0; // 搜索区间1到x因为1的平方根是1x的平方根不大于x long left 1; long right x; while (left right) { // 向上取整避免死循环若向下取整可能导致left无法到达right long mid left (right - left 1) / 2; if (mid * mid x) { left mid; // mid满足条件尝试找更大的满足条件的数 } else { right mid - 1; // mid不满足条件缩小右边界 } } // 循环结束后leftright即为x的算术平方根的整数部分 return (int) left; } }题目 4LeetCode 852 / 162 山峰数组、寻找峰值这道题考察二分查找的核心本质——二段性而非数组有序。山峰数组的特点是前半部分严格递增后半部分严格递减峰值就是数组中最大的元素也是递增和递减的分界点利用这一二段性可快速用二分找到峰值。class Solution { public int peakIndexInMountainArray(int[] arr) { // 搜索区间1到arr.length-2峰值不可能在数组两端避免越界 int left 1; int right arr.length - 2; while (left right) { int mid left (right - left 1) / 2; // 若mid位置元素大于前一个元素说明在递增段峰值在右侧 if (arr[mid] arr[mid - 1]) { left mid; } else { // 否则在递减段峰值在左侧 right mid - 1; } } // 循环结束后leftright即为峰值下标 return left; } }题目 5LeetCode 153 寻找旋转排序数组中的最小值旋转排序数组是二分查找的经典变形场景这类数组是由有序数组旋转得到例如[0,1,2,4,5,6,7]旋转后得到[4,5,6,7,0,1,2]其依然具有二段性一部分是有序递增区间另一部分也是有序递增区间且前半段区间的所有元素都大于后半段区间的所有元素利用这一特点可快速找到最小值。class Solution { public int findMin(int[] nums) { int left 0; int right nums.length - 1; // 以数组最右侧元素为基准判断mid所在的区间 int x nums[right]; while (left right) { int mid left (right - left) / 2; // 若mid元素大于基准x说明mid在左半段较大元素区间最小值在右侧 if (nums[mid] x) { left mid 1; } else { // 否则mid在右半段较小元素区间最小值在左侧或当前位置 right mid; } } // 循环结束后leftright即为最小值下标 return nums[left]; } }五、二分查找核心总结必背二分的本质不是有序而是二段性这是最核心的知识点只要能将搜索区间明确划分为“满足条件”和“不满足条件”两部分且两部分边界清晰无论数组是否整体有序都能使用二分查找。循环条件核心区分找确定值如LeetCode 704left right需判断每一个位置的元素避免遗漏找边界/峰值/最小值如左右边界、山峰数组left right循环结束后left与right重合再判断该位置是否符合要求。mid 写法避坑关键向下取整left (right-left)/2适用于找左边界、确定值查找等场景向上取整left (right-left1)/2适用于找右边界、峰值等场景避免出现死循环。区间收缩灵活运用右移left mid 1当mid位置元素不满足目标条件且目标在右侧区间时使用左移right mid - 1当mid位置元素不满足目标条件且目标在左侧区间时使用保留 midleftmid或rightmid当mid位置元素可能是目标值如边界、峰值需要保留该位置继续查找时使用。六、学习建议先把704 基础二分写熟这是所有二分题的基础先熟练掌握基础模板理解循环条件、mid写法、区间收缩的逻辑不要急于刷难题再练34 左右边界左右边界是二分变形的核心多手推几遍区间变化过程理解“向下取整/向上取整”和“保留mid”的原因避免死记模板然后按顺序刷题35插入位置→ 69平方根→ 852山峰数组→ 153旋转数组最小值→ 162寻找峰值循序渐进巩固不同场景的应用每道题自己手推一遍区间变化不要死记模板二分的难点在于边界处理手推区间变化如left、right、mid的每次取值能快速理解收缩逻辑避免出错。最后提醒二分查找看似简单实则细节为王很多人刷题时出错不是不会模板而是忽略了边界条件和区间收缩的细节。只要理解区间怎么缩掌握二段性的核心所有二分题都是换汤不换药多练、多推、多总结就能轻松应对面试中的各类二分考点。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2413627.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…