二分查找算法详解:在排序数组中查找元素的首尾位置
二分查找算法通过不断减半搜索区间实现 O(log n) 时间复杂度。内容涵盖基础二分查找定位目标值,以及扩展应用:在有序数组中查找目标值的起始和结束位置。核心在于调整左右边界指针的移动逻辑,分别寻找左边界和右边界,避免死循环并处理边界条件。提供 C++ 代码实现及模板,适用于排序数据的高效检索场景。

二分查找算法通过不断减半搜索区间实现 O(log n) 时间复杂度。内容涵盖基础二分查找定位目标值,以及扩展应用:在有序数组中查找目标值的起始和结束位置。核心在于调整左右边界指针的移动逻辑,分别寻找左边界和右边界,避免死循环并处理边界条件。提供 C++ 代码实现及模板,适用于排序数据的高效检索场景。

二分查找(Binary Search),也称为折半查找,是一种高效的有序数组查找算法。其核心思想是通过不断将搜索区间减半,快速缩小目标值的可能范围,最终找到目标值或确定其不存在。该算法的时间复杂度为 O(log n),远优于顺序查找的 O(n),在处理大规模有序数据时优势显著。
题目描述: 给定一个 n 个元素有序的(升序)整型数组 nums 和一个目标值 target ,写一个函数搜索 nums 中的 target,如果目标值存在返回下标,否则返回 -1。
题目示例: 输入:nums = [-1,0,3,5,9,12], target = 9 输出:4 解释:9 出现在 nums 中并且下标为 4
class Solution {
public:
int search(vector<int>& nums, int target) {
int left = 0, right = nums.size() - 1;
while (left <= right) {
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;
}
};
int left = 0, right = nums.size() - 1;
while (left <= right) {
int mid = left + (right - left) / 2; // 防止数据溢出
if (条件判断) left = mid + 1;
else if (条件判断) right = mid - 1;
else return 结果;
}
题目描述: 给你一个按照非递减顺序排列的整数数组 nums,和一个目标值 target。请你找出给定目标值在数组中的开始位置和结束位置。 如果数组中不存在目标值 target,返回 [-1, -1]。
题目示例: 输入:nums = [5,7,7,8,8,10], target = 8 输出:[3,4]
算法思路: 使用二分思想,根据数据的性质,在某种判断条件下将区间一分为二,然后舍去其中一个区间,再在另一个区间内查找。 用 x 表示该元素,resLeft 表示左边界,resRight 表示右边界。
寻找左边界思路: 以左边界划分的两个区间的特点:
关于 mid 的落点,分为两种情况:
注意:这里找中间元素需要向下取整。因为后续移动左右指针的时候,右指针 right = mid 可能会原地踏步(例如剩下 1,2 两个元素,left=1, right=2, mid=2,更新后值不变会陷入死循环)。因此当 right = mid 的时候,要向下取整。
寻找右边界思路: 用 resRight 表示右边界,注意到右边界的特点:
关于 mid 的落点,分为两种情况:
注意:这里找中间元素需要向上取整。因为后续移动左右指针的时候,左指针 left = mid 可能会原地踏步(例如剩下 1,2 两个元素,left=1, right=2, mid=1,更新后值不变会陷入死循环)。因此当 left = mid 的时候,要向上取整。
C++ 算法代码:
class Solution {
public:
vector<int> searchRange(vector<int>& nums, int target) {
int begin = 0, end = 0, mid = 0;
if (nums.empty()) {
return {-1, -1};
}
// 1. 查找区间左端点
int left = 0, right = nums.size() - 1;
while (left < right) {
mid = left + (right - left) / 2;
if (nums[mid] < target) left = mid + 1;
else right = mid;
}
if (nums[left] == target) begin = left;
else return {-1, -1};
// 2. 查找区间右端点
left = 0, right = nums.size() - 1;
while (left < right) {
mid = left + (right - left + 1) / 2; // 向上取整
if (nums[mid] <= target) left = mid;
else right = mid - 1;
}
if (nums[right] == target) end = right;
else return {-1, -1};
{begin, end};
}
};
本文通过两道经典题目讲解了二分查找的基础应用及变体。掌握二分查找的关键在于正确维护左右边界以及处理边界条件,避免死循环。对于有序数组的检索问题,优先考虑二分查找以降低时间复杂度。

微信公众号「极客日志」,在微信中扫描左侧二维码关注。展示文案:极客日志 zeeklog
使用加密算法(如AES、TripleDES、Rabbit或RC4)加密和解密文本明文。 在线工具,加密/解密文本在线工具,online
将字符串编码和解码为其 Base64 格式表示形式即可。 在线工具,Base64 字符串编码/解码在线工具,online
将字符串、文件或图像转换为其 Base64 表示形式。 在线工具,Base64 文件转换器在线工具,online
将 Markdown(GFM)转为 HTML 片段,浏览器内 marked 解析;与 HTML转Markdown 互为补充。 在线工具,Markdown转HTML在线工具,online
将 HTML 片段转为 GitHub Flavored Markdown,支持标题、列表、链接、代码块与表格等;浏览器内处理,可链接预填。 在线工具,HTML转Markdown在线工具,online
通过删除不必要的空白来缩小和压缩JSON。 在线工具,JSON 压缩在线工具,online