当前位置: 首页 > article >正文

leetcode74:搜索二维矩阵

原题地址:74. 搜索二维矩阵 - 力扣(LeetCode)

题目描述

给你一个满足下述两条属性的 m x n 整数矩阵:

  • 每行中的整数从左到右按非严格递增顺序排列。
  • 每行的第一个整数大于前一行的最后一个整数。

给你一个整数 target ,如果 target 在矩阵中,返回 true ;否则,返回 false 。

示例 1:

输入:matrix = [[1,3,5,7],[10,11,16,20],[23,30,34,60]], target = 3
输出:true

示例 2:

输入:matrix = [[1,3,5,7],[10,11,16,20],[23,30,34,60]], target = 13
输出:false

解题思路

题目要求在一个二维矩阵中查找一个目标值,矩阵满足以下特性:

  1. 每一行中的整数从左到右按升序排列。
  2. 每一行的第一个整数大于上一行的最后一个整数。

解题可以分两步:

  1. 确定目标所在的行
    • 使用二分查找在第一列中查找,确定目标可能所在的行索引。
    • 如果目标值小于第一行的第一个元素或大于最后一行的第一个元素,返回 false
  2. 在该行中查找目标值
    • 在确定的行中再用一次二分查找,查看目标值是否存在。

这种方法利用了矩阵的有序性,避免了直接遍历整个矩阵,时间复杂度更低。

源码实现

class Solution {
    public boolean searchMatrix(int[][] matrix, int target) {
        // 使用二分查找第一列,找到目标值可能存在的行索引
        int rowIndex = binarySearchFirstColumn(matrix, target);
        if (rowIndex < 0) { // 如果返回的行索引小于0,说明目标值不在矩阵中
            return false;
        }
        // 在找到的行中再次使用二分查找,确认目标值是否存在
        return binarySearchRow(matrix[rowIndex], target);
    }

    // 在矩阵的第一列中使用二分查找,返回目标值可能存在的行索引
    public int binarySearchFirstColumn(int[][] matrix, int target) {
        int low = -1, high = matrix.length - 1; // 初始化low为-1,high为最后一行索引
        while (low < high) {
            // 防止溢出的写法,计算中点位置
            int mid = (high - low + 1) / 2 + low;
            if (matrix[mid][0] <= target) { // 如果当前行的第一个元素小于或等于目标值
                low = mid; // 目标值可能在该行或更低的行
            } else {
                high = mid - 1; // 向上搜索
            }
        }
        return low; // 返回目标值可能存在的行索引
    }

    // 在给定行中使用二分查找,判断目标值是否存在
    public boolean binarySearchRow(int[] row, int target) {
        int low = 0, high = row.length - 1; // 初始化low为0,high为行的最后一个元素索引
        while (low <= high) {
            int mid = (high - low) / 2 + low; // 防止溢出的写法
            if (row[mid] == target) { // 如果中间值等于目标值
                return true; // 找到目标值
            } else if (row[mid] > target) { // 如果中间值大于目标值
                high = mid - 1; // 向左搜索
            } else { // 如果中间值小于目标值
                low = mid + 1; // 向右搜索
            }
        }
        return false; // 未找到目标值
    }
}

复杂度分析

  1. 时间复杂度

    • 第一步(二分查找第一列)
      • 矩阵有 m 行,查找第一列的时间复杂度为 O(log m)
    • 第二步(二分查找一行)
      • 每行有 n 列,查找该行的时间复杂度为 O(log n)
    • 总时间复杂度:O(log m + log n),可以简化为 O(log (m * n))
  2. 空间复杂度

    • 算法使用了常数级的额外空间,没有开辟新的数据结构。
    • 空间复杂度为 O(1)

http://www.kler.cn/a/443832.html

相关文章:

  • TCP与UDP的端口连通性
  • 使用Vscode+EIDE+Jlink开发STM32环境配置教程
  • 单调栈基础用法
  • CCF-GESP 等级考试 2023年9月认证C++一级真题解析
  • LeetCode题练习与总结:预测赢家--486
  • SYD881X RTC定时器事件在调用timeAppClockSet后会出现比较大的延迟
  • 从 PDF 到 Word:一个简单的 PythonGUI转换器
  • 请给我详细讲解vue.config.js的配置内容
  • React状态管理常见面试题目(二)
  • Vue前端开发-数据缓存
  • K-Means 聚类:数据挖掘的瑞士军刀
  • 将java项目部署到linux
  • Selenium 深度解析:自动化浏览器操作的利器
  • PPT中添加多个图片
  • 解决echarts图宽度自适应问题,设置100%宽度显示100px
  • UDP网络编程套接
  • Java.10--IO流
  • 修改openjdk17 java/lang/String.java 类源码,增加一个native本地方法打印固定字符串功能
  • 图书馆管理系统(一)基于jquery、ajax
  • Linux 显示系统活动进程状态命令 ps 详细介绍
  • 如何有效修复ffmpeg.dll错误:一站式解决方案指南
  • Linux dd 命令详解:工作原理与实用指南(C/C++代码实现)
  • 单节点calico性能优化
  • springboot444新冠物资管理系统的设计与实现(论文+源码)_kaic
  • Databend 产品月报(2024年11月)
  • 【深度学习之三】FPN与PAN网络详解