算法练习题18——leetcode240搜索二维矩阵||(二分)

题目描述

编写一个高效的算法来搜索 m xn 矩阵 matrix 中的一个目标值 target 。该矩阵具有以下特性:

每行的元素从左到右升序排列。

每列的元素从上到下升序排列。

代码

复制代码
class Solution {
    public boolean searchMatrix(int[][] matrix, int target) {
        for(int[] row:matrix){
            int index = search(row,target);
            if(index>=0){
                return true;
            }
        }
        return false;
    }
    public int search(int[] nums, int target) {
        int low = 0, high = nums.length - 1; // 初始化指针
        while (low <= high) {
            int mid = (high - low) / 2 + low; // 计算中点,防止整数溢出
            int num = nums[mid]; // 获取中点值
            if (num == target) { // 找到目标值
                return mid;
            } else if (num > target) { // 目标值在左半部分
                high = mid - 1;
            } else { // 目标值在右半部分
                low = mid + 1;
            }
        }    
    return -1; // 没有找到目标值
    }
}

复杂度分析

  • 时间复杂度:O(mlogn)。对一行使用二分查找的时间复杂度为 O(logn),最多需要进行 m 次二分查找。

  • 空间复杂度:O(1)。

示例和解释

假设我们有一个升序的数组 nums = [1, 3, 5, 7, 9, 11],目标值 target = 7。让我们看看这个算法如何找到目标值:

  1. 初始状态

    • 数组:[1, 3, 5, 7, 9, 11]
    • low = 0high = 5(数组的长度为 6,所以索引范围是 05
  2. 第一次迭代

    • 计算中点索引:mid = (high - low) / 2 + low = (5 - 0) / 2 + 0 = 2
    • 中点值:nums[mid] = nums[2] = 5
    • 比较:5 < 7(中点值小于目标值)
    • 更新指针:将 low 指针移动到 mid + 1 = 3high 保持不变(5
  3. 第二次迭代

    • 新的范围:[7, 9, 11](索引 35
    • 计算中点索引:mid = (high - low) / 2 + low = (5 - 3) / 2 + 3 = 4
    • 中点值:nums[mid] = nums[4] = 9
    • 比较:9 > 7(中点值大于目标值)
    • 更新指针:将 high 指针移动到 mid - 1 = 3low 保持不变(3
  4. 第三次迭代

    • 新的范围:[7](索引 33
    • 计算中点索引:mid = (high - low) / 2 + low = (3 - 3) / 2 + 3 = 3
    • 中点值:nums[mid] = nums[3] = 7
    • 比较:7 == 7(中点值等于目标值)
    • 找到目标值,返回 mid = 3

结果

目标值 7 在数组中的索引是 3,所以函数返回 3

另一个示例

假设我们有同样的数组 nums = [1, 3, 5, 7, 9, 11],但目标值 target = 4,这个值不在数组中。我们看看算法如何工作:

  1. 初始状态

    • 数组:[1, 3, 5, 7, 9, 11]
    • low = 0high = 5
  2. 第一次迭代

    • 计算中点索引:mid = (high - low) / 2 + low = (5 - 0) / 2 + 0 = 2
    • 中点值:nums[mid] = nums[2] = 5
    • 比较:5 > 4(中点值大于目标值)
    • 更新指针:将 high 指针移动到 mid - 1 = 1low 保持不变(0
  3. 第二次迭代

    • 新的范围:[1, 3](索引 01
    • 计算中点索引:mid = (high - low) / 2 + low = (1 - 0) / 2 + 0 = 0
    • 中点值:nums[mid] = nums[0] = 1
    • 比较:1 < 4(中点值小于目标值)
    • 更新指针:将 low 指针移动到 mid + 1 = 1high 保持不变(1
  4. 第三次迭代

    • 新的范围:[3](索引 11
    • 计算中点索引:mid = (high - low) / 2 + low = (1 - 1) / 2 + 1 = 1
    • 中点值:nums[mid] = nums[1] = 3
    • 比较:3 < 4(中点值小于目标值)
    • 更新指针:将 low 指针移动到 mid + 1 = 2
  5. 结束条件

    • 现在 low = 2high = 1,满足 low > high,循环结束。
    • 目标值 4 不在数组中,返回 -1
相关推荐
兔兔爱学习兔兔爱学习11 分钟前
Spring Al学习7:ImageModel
java·学习·spring
lang201509281 小时前
Spring远程调用与Web服务全解析
java·前端·spring
m0_564264182 小时前
IDEA DEBUG调试时如何获取 MyBatis-Plus 动态拼接的 SQL?
java·数据库·spring boot·sql·mybatis·debug·mybatis-plus
崎岖Qiu2 小时前
【设计模式笔记06】:单一职责原则
java·笔记·设计模式·单一职责原则
Hello.Reader2 小时前
Flink ExecutionConfig 实战并行度、序列化、对象重用与全局参数
java·大数据·flink
前端炒粉3 小时前
18.矩阵置零(原地算法)
javascript·线性代数·算法·矩阵
熊小猿3 小时前
在 Spring Boot 项目中使用分页插件的两种常见方式
java·spring boot·后端
paopaokaka_luck3 小时前
基于SpringBoot+Vue的助农扶贫平台(AI问答、WebSocket实时聊天、快递物流API、协同过滤算法、Echarts图形化分析、分享链接到微博)
java·vue.js·spring boot·后端·websocket·spring
老华带你飞3 小时前
机器人信息|基于Springboot的机器人门户展示系统设计与实现(源码+数据库+文档)
java·数据库·spring boot·机器人·论文·毕设·机器人门户展示系统
AI Chen3 小时前
【矩阵分析与应用】【第1章 矩阵与线性方程组】【1.6.2.2 迹的循环置换性质】
矩阵·