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

腾讯面试手撕题:返回行递增有序矩阵第k小的元素

题目

给定一个n行n列的矩阵,这个矩阵的每一行是递增有序的,求这个矩阵中第k小的元素。

解答

优解

基于二分查找和按行统计小于等于目标值的元素个数。算法的时间复杂度为O(nlognlogD),其中D是矩阵中元素值域的范围(即最大值与最小值的差),空间复杂度为O(1)(不包括输入矩阵)。

算法描述
  1. 确定值域范围

    • 计算矩阵中的最小值 min_val:取所有行首元素(即每行的第一个元素)的最小值,即 min_val = min(matrix[i][0] for i in range(n))

    • 计算矩阵中的最大值 max_val:取所有行尾元素(即每行的最后一个元素)的最大值,即 max_val = max(matrix[i][n-1] for i in range(n))

    • 初始化二分查找的左边界 left = min_val,右边界 right = max_val

  2. 二分查找第 k 小元素

    • 当 left < right 时,执行循环:

      • 计算中间值 mid = (left + right) // 2(整数除法)。

      • 统计矩阵中小于等于 mid 的元素个数 count

        • 对于每一行 i(因为行是递增有序),使用二分查找找到该行中最后一个小于等于 mid 的元素的列索引 j(即最大的 j 满足 matrix[i][j] <= mid)。则该行中小于等于 mid 的元素个数为 j+1(列索引从 0 开始)。

        • 对所有行的结果求和,得到 count

      • 如果 count >= k,则第 k 小元素小于等于 mid,设置 right = mid

      • 否则,第 k 小元素大于 mid,设置 left = mid + 1

    • 循环结束后,left 即为第 k 小元素的值。

算法正确性
  • 该算法通过二分查找值域,逐步缩小第 k 小元素可能存在的范围。

  • 在每次迭代中,计算 count(mid)(小于等于 mid 的元素个数)与 k 比较,可以确定第 k 小元素位于左半区间还是右半区间。

  • 最终,left 会收敛到矩阵中的一个实际元素,且满足是第 k 小元素(详见示例验证)。

时间复杂度分析
  • 确定 min_val 和 max_val 需要O(n)时间。

  • 二分查找的迭代次数为O(logD),其中D=max_val−min_val。

  • 在每次迭代中,统计 count(mid) 需要对每一行进行一次二分查找(每行时间复杂度O(logn)),因此每次迭代的时间复杂度为O(nlogn)

  • 总时间复杂度为O(nlognlogD)

示例说明

考虑一个 2×2 矩阵:

\begin{bmatrix} 1 & 3\\ 2& 5 \end{bmatrix}

元素集合为 {1,2,3,5},排序后为 [1,2,3,5]。

  • 求第 k=2 小元素:

    • min_val = min(1, 2) = 1max_val = max(3, 5) = 5left = 1right = 5

    • 第一次迭代:mid = (1 + 5) // 2 = 3count(<=3)

      • 第一行:元素 [1,3],1≤3,3≤3,个数为 2。

      • 第二行:元素 [2,5],2≤3,5>3,个数为 1。

      • count = 2 + 1 = 3 >= 2,设置 right = 3

    • 第二次迭代:left = 1right = 3mid = (1 + 3) // 2 = 2count(<=2)

      • 第一行:1≤2,3>2,个数为 1。

      • 第二行:2≤2,5>2,个数为 1。

      • count = 1 + 1 = 2 >= 2,设置 right = 2

    • 第三次迭代:left = 1right = 2mid = (1 + 2) // 2 = 1count(<=1)

      • 第一行:1≤1,3>1,个数为 1。

      • 第二行:2>1,个数为 0。

      • count = 1 < 2,设置 left = 1 + 1 = 2

    • 循环结束,left = 2,返回 2(正确,第 2 小元素是 2)。

代码实现
def kthSmallest(matrix, k):n = len(matrix)# 初始化二分查找的边界left = matrix[0][0]              # 最小值:矩阵左上角right = matrix[n-1][n-1]         # 最大值:矩阵右下角# 二分查找while left < right:mid = (left + right) // 2count = 0                    # 统计小于等于mid的元素个数col = n - 1                 # 从每行的末尾开始检查# 遍历每一行for i in range(n):# 在当前行中,从右向左找到第一个小于等于mid的元素while col >= 0 and matrix[i][col] > mid:col -= 1count += (col + 1)       # 该行中小于等于mid的元素个数# 调整二分查找边界if count >= k:right = midelse:left = mid + 1return left# 示例测试
if __name__ == "__main__":matrix1 = [[1, 3], [2, 5]]print(kthSmallest(matrix1, 2))  # 输出: 2matrix2 = [[1, 5, 9], [10, 11, 13], [12, 13, 15]]print(kthSmallest(matrix2, 4))  # 输出: 10matrix3 = [[-5]]print(kthSmallest(matrix3, 1))  # 输出: -5

暴力解

忽略行递增条件,直接全排序:

def kth_smallest(matrix, k):n = len(matrix)# 提取所有元素到一维列表flat_list = []for i in range(n):for j in range(n):flat_list.append(matrix[i][j])# 排序列表flat_list.sort()# 返回第k小的元素(k从1开始,索引为k-1)return flat_list[k-1]
http://www.xdnf.cn/news/742357.html

相关文章:

  • 【教学类-36-10】20250531蝴蝶图案描边,最适合大小(一页1图1图、2图图案不同、2图图案相同对称)
  • C++ 重载(Overload)、重写(Override)、隐藏(Hiding) 的区别
  • LiquiGen流体导入UE
  • STM32 HAL库函数学习 CRC篇
  • Linux系统编程之共享内存
  • 在QT中,利用charts库绘制FFT图形
  • MAC软件游戏打开提示已损坏
  • MATLAB实战:机器学习分类回归示例
  • 【MFC】如何设置让exe的控制台不会跟着exe退出而退出
  • C++中指针常量和常量指针的区别
  • 【设计模式-4.6】行为型——状态模式
  • [蓝桥杯]拉马车
  • L56.【LeetCode题解】 电话号码的字母组合
  • 触发器与存储过程详解
  • Mybatis-Plus简单介绍
  • 鸿蒙HarmonyOS (React Native)的实战教程
  • Java后端技术栈问题排查实战:Spring Boot启动慢、Redis缓存击穿与Kafka消费堆积
  • 【Java学习笔记】内部类(重点)
  • 数据结构:时间复杂度(Time Complexity)和空间复杂度(Space Complexity)
  • Typescript学习教程,从入门到精通,TypeScript 配置管理与编译器详解(19)
  • Rust 配置解析`serde` + `toml`
  • 华为OD机试真题——找出两个整数数组中同时出现的整数(2025A卷:100分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
  • 【HTML】基础学习【数据分析全栈攻略:爬虫+处理+可视化+报告】
  • MySQL事务与锁机制详解:确保数据一致性的关键【MySQL系列】
  • 005 flutter基础,初始文件讲解(4)
  • leetcode付费题 353. 贪吃蛇游戏解题思路
  • 实现MPC钱包
  • [蓝桥杯]阶乘求值【省模拟赛】
  • Thinkphp6实现websocket
  • Spring Boot养老院管理系统源码分享