搜索二维矩阵

题目理解

matrix 是一个 行排序矩阵,即:

• 每一行的元素从 左到右递增

• 每一行的 最右元素 一定大于等于该行所有元素,且小于等于下一行的所有元素(如果有下一行)。

目标:判断 target 是否在 matrix 中

解法思路

我们采用 两阶段二分查找 进行优化:

  1. 第一步 :遍历每一行,找到可能包含 target 的行

• 如果 target 大于 当前行的最后一个元素,则 target 只能在后面的行,继续下一行。

• 否则,该行 可能包含 target ,进入 第二步

  1. 第二步 :在该行使用 二分查找 搜索 target。

• 如果找到,返回 true。

• 否则,返回 false(因为 target 只可能在当前行,不在其他行)。

代码解析

cpp 复制代码
class Solution {
public:
    bool searchMatrix(vector<vector<int>>& matrix, int target) {
        int n = matrix.size();     // 矩阵行数
        int m = matrix[0].size();  // 矩阵列数

        for (int i = 0; i < n; i++) {  // 遍历每一行
            if (target > matrix[i][m - 1]) continue; // `target` 大于该行最后一个元素,跳过
            
            // 二分查找
            int left = 0, right = m - 1;
            while (left <= right) {
                int mid = left + (right - left) / 2;  // 防止溢出

                if (matrix[i][mid] == target) return true; // 找到 `target`
                else if (matrix[i][mid] > target) right = mid - 1; // 缩小右边界
                else left = mid + 1;  // 缩小左边界
            }
            break;  // 只可能在这一行找到,找不到就退出
        }
        return false;  // 遍历完所有可能行,仍找不到
    }
};

详细运行步骤

示例 1

复制代码
vector<vector<int>> matrix = {
    {1, 3, 5, 7},
    {10, 11, 16, 20},
    {23, 30, 34, 60}
};
int target = 3;

矩阵表示:

复制代码
1   3   5   7
10  11  16  20
23  30  34  60

执行过程

第一步:寻找目标行

• 遍历第 0 行,最后元素 7:

• target = 3 7,可能在当前行 ,进入 二分查找

第二步:二分查找

• left = 0, right = 3, mid = (0+3)/2 = 1

• matrix[0][1] = 3 == target,返回 true。

最终结果:true

示例 2

复制代码
vector<vector<int>> matrix = {
    {1, 3, 5, 7},
    {10, 11, 16, 20},
    {23, 30, 34, 60}
};
int target = 13;

第一步:寻找目标行

第 0 行 :7 < 13,跳过。

第 1 行 :20 13,进入 二分查找

第二步:二分查找

• left = 0, right = 3, mid = 1

• matrix[1][1] = 11 < target → left = mid + 1 = 2

• left = 2, right = 3, mid = 2

• matrix[1][2] = 16 > target → right = mid - 1 = 1

• left > right,找不到,返回 false。

最终结果:false

示例 3

复制代码
vector<vector<int>> matrix = {
    {1, 3, 5, 7},
    {10, 11, 16, 20},
    {23, 30, 34, 60}
};
int target = 60;

第一步:寻找目标行

第 0 行 :7 < 60,跳过。

第 1 行 :20 < 60,跳过。

第 2 行 :60 60,进入 二分查找

第二步:二分查找

• left = 0, right = 3, mid = 1

• matrix[2][1] = 30 < target → left = mid + 1 = 2

• left = 2, right = 3, mid = 2

• matrix[2][2] = 34 < target → left = mid + 1 = 3

• left = 3, right = 3, mid = 3

• matrix[2][3] = 60 == target,返回 true。

最终结果:true

复杂度分析

时间复杂度

  1. 遍历行:O(N)

最坏情况,可能要遍历所有行。

最优情况,第一行就满足条件。

  1. 二分查找:O(log M)

• 在选定的一行内,二分查找 O(log M)

总复杂度

通常 M 很大时,log M 远小于 N,所以二分查找优化明显。

空间复杂度

• 只使用了常数变量 left、right、mid,不需要额外的数组。

优化建议

当前算法在最坏情况下需要遍历所有行。可以使用 一次性二分查找 代替两次:

  1. 把矩阵视为一个一维数组,索引映射:

  2. 直接对整个矩阵二分查找

cpp 复制代码
class Solution {
public:
    bool searchMatrix(vector<vector<int>>& matrix, int target) {
        int n = matrix.size(), m = matrix[0].size();
        int left = 0, right = n * m - 1;

        while (left <= right) {
            int mid = left + (right - left) / 2;
            int val = matrix[mid / m][mid % m];  // 计算对应的行列索引

            if (val == target) return true;
            else if (val > target) right = mid - 1;
            else left = mid + 1;
        }
        return false;
    }
};

时间复杂度:O(log (N * M))

空间复杂度:O(1)

这个方法比你的方法 更快 ,适用于 行列均递增的矩阵

总结

你的方法

两步二分(O(N + log M))。

适用于行递增的矩阵

优化方案

一维二分(O(log (N*M)),更快)。

适用于行列均递增的矩阵

👉 如果 matrix 行列都递增,建议用 O(log(N*M)) 版本!

相关推荐
老鱼说AI4 小时前
算法基础教学第一步:数据结构
数据结构·python·算法
Jing_Rainbow4 小时前
【LeetCode Hot100 刷题日记(19/100)】54. 螺旋矩阵 —— 数组、矩阵、模拟、双指针、层序遍历🌀
算法·面试·程序员
地平线开发者5 小时前
征程 6 | linear 高精度输出配置方式
算法·自动驾驶
小尧嵌入式5 小时前
C++基础语法总结
开发语言·c++·stm32·单片机·嵌入式硬件·算法
white-persist5 小时前
【攻防世界】reverse | IgniteMe 详细题解 WP
c语言·汇编·数据结构·c++·python·算法·网络安全
稚辉君.MCA_P8_Java5 小时前
Gemini永久会员 归并排序(Merge Sort) 基于分治思想(Divide and Conquer)的高效排序算法
java·linux·算法·spring·排序算法
地平线开发者6 小时前
征程 6 | QAT 新版 qconfig 量化模板使用教程
算法·自动驾驶
多恩Stone6 小时前
【ModelScope-1】数据集稀疏检出(Sparse Checkout)来下载指定目录
人工智能·python·算法·aigc
山峰哥6 小时前
沉浸式翻译插件深度评测:打破语言壁垒的黑科技利器
数据结构·科技·算法·编辑器·办公
AI脚下的巨人7 小时前
机器人逆运动学:从SVD到IK算法
算法·机器人