刷题 二分查找

二分查找

二分查找的本质就是 缩小有效范围

需要注意:

  • int mid = (left + right) / 2; int mid = left + (right - left) / 2; 防止溢出

hot100 - 二分查找

⭐️35. 搜索插入位置

找到第一个大于等于 target 的值

cpp 复制代码
class Solution {
public:
    // 目标: 找到第一个大于等于 target 的值
    int searchInsert(vector<int>& nums, int target) {
        int left = 0, right = nums.size() - 1;
        while (left <= right) { // 区间不为空
            // 区间缩减的循环不变量
            // nums[left - 1] < target
            // nums[right + 1] >= target
            int mid = left + (right - left) / 2;
            if (nums[mid] < target) {
                left = mid + 1;     // 缩小范围到 [mid + 1, right]
            } else {
                right = mid - 1;    // 缩小范围到 [left, mid - 1]
            }
        }
        return left;
    }
};
cpp 复制代码
class Solution {
public:
    // 目标: 找到第一个大于等于 target 的值
    int searchInsert(vector<int>& nums, int target) {
        int left = 0, right = nums.size();
        while (left < right) { // 区间不为空
            // 区间缩减的循环不变量
            // nums[left - 1] < target
            // nums[right] >= target
            int mid = left + (right - left) / 2;
            if (nums[mid] < target) {
                left = mid + 1;     // 缩小范围到 [mid + 1, right]
            } else {
                right = mid;    // 缩小范围到 [left, mid - 1]
            }
        }
        return left;
    }
};

74. 搜索二维矩阵

cpp 复制代码
class Solution {
public:
    // 二分法的核心:缩小有效区间
    bool searchMatrix(vector<vector<int>>& matrix, int target) {
        int n = matrix.size(), m = matrix[0].size();
        if (target < matrix[0][0] || target > matrix[n - 1][m - 1]) return false;

        int left = 0, right = n - 1, row = 0, col = 0;
        
        // 二分查找,先确定在哪一行
        while (left <= right) {
            // 循环不变量
            // nums[left-1][m-1] < target
            // nums[right+1][m-1] > target
            row = left + (right - left) / 2;
            if (matrix[row][m - 1] == target) {
                return true;
            } else if (matrix[row][m - 1] < target) {
                left = row + 1;
            } else {
                right = row - 1;
            }
        }
        // 循环结束
        row = left;

        // 二分查找,再确定在哪一列
        left = 0, right = m - 1;
        while (left <= right) {
            // 循环不变量
            // nums[row][left-1] < target
            // nums[row][right+1] > target
            col = left + (right - left) / 2;
            if (matrix[row][col] == target) {
                return true;
            } else if (matrix[row][col] < target) {
                left = col + 1;
            } else {
                right = col - 1;
            }
        }
        return false;
    }
};

⭐️⭐️⭐️162. 寻找峰值


cpp 复制代码
class Solution {
public:
    // 首先我们可以判定,数组中必然包含至少一个峰值元素
    // 峰值元素的特征 nums[i] > nums[i-1] nums[i] > nums[i + 1]
    // 相当于找极大值点
    int findPeakElement(vector<int>& nums) {
        int n = nums.size();
        int left = 0, right = n - 1;
        while (left < right) {
            int mid = left + (right - left) / 2;
            if ((mid == 0 || nums[mid] > nums[mid - 1]) && (mid != n - 1 && nums[mid] < nums[mid + 1])) {
                left = mid + 1;
            } else if ((mid != 0 && nums[mid] < nums[mid - 1]) && (mid == n - 1 || nums[mid] > nums[mid + 1])) {
                right = mid - 1;
            } else if ((mid == 0 || nums[mid] > nums[mid - 1]) && (mid == n - 1 || nums[mid] > nums[mid + 1])) {
                return mid;
            } else {
                // 以下两种都可以
                left = mid + 1;
                // right = mid - 1;
            }
        }
        return left;
    }
};

代码优化

严格证明,简洁写法(Python/Java/C++/C/Go)

cpp 复制代码
class Solution {
public:
    // 首先我们可以确定必然有一个峰值, 因此我们只需要进行区间缩减即可
    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]) {    // 如果 nums[mid] < nums[mid + 1],说明右侧有峰值
                left = mid + 1;
            } else {                            // nums[mid] > nums[mid + 1], 说明左侧有峰值
                right = mid;
            }
        }
        // left == right 时即为峰值位置
        return left;
    }
};

33. 搜索旋转排序数组

看清楚题意,是将一个升序序列旋转一下,不是两个升序序列拼在一起

cpp 复制代码
class Solution {
public:
    int search(vector<int>& nums, int target) {
        int n = nums.size(), left = 0, right = n - 1;
        while (left <= right) {
            int mid = left + (right - left) / 2;
            if (target == nums[mid]) {
                return mid;
            } else {
                if (nums[mid] >= nums[0]) { // 在第一个上升区间 nums[mid] >= nums[0]
                    if (target > nums[mid] || target < nums[0]) left = mid + 1;
                    else right = mid - 1;
                } else { // 在第二个上升区间 nums[mid] < nums[0]
                    if (target < nums[mid] || target > nums[n - 1]) right = mid - 1;
                    else left = mid + 1;
                }
            }
        }
        return -1;
    }
};

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

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

4. 寻找两个正序数组的中位数

相关推荐
笑鸿的学习笔记5 分钟前
qt-C++语法笔记之Stretch与Spacer的关系分析
c++·笔记·qt
hardStudy_h20 分钟前
C++——内联函数与Lambda表达式
开发语言·jvm·c++
ZZZS05161 小时前
stack栈练习
c++·笔记·学习·算法·动态规划
位东风1 小时前
【c++学习记录】状态模式,实现一个登陆功能
c++·学习·状态模式
雷羿 LexChien3 小时前
C++内存泄漏排查
开发语言·c++
嘉小华4 小时前
CMake 完全指南:第一章 - 构建的烦恼 - 为什么需要CMake?
c++
Feliz Da Vida4 小时前
[代码学习] c++ 通过H矩阵快速生成图像对应的mask
c++·学习
无聊的小坏坏5 小时前
单调栈通关指南:从力扣 84 到力扣 42
c++·算法·leetcode
YOLO大师6 小时前
华为OD机试 2025B卷 - 小明减肥(C++&Python&JAVA&JS&C语言)
c++·python·华为od·华为od机试·华为od2025b卷·华为机试2025b卷·华为od机试2025b卷
看到我,请让我去学习7 小时前
OpenCV编程- (图像基础处理:噪声、滤波、直方图与边缘检测)
c语言·c++·人工智能·opencv·计算机视觉