有序矩阵中第K小的元素

给你一个 n x n矩阵 matrix ,其中每行和每列元素均按升序排序,找到矩阵中第 k 小的元素。

请注意,它是 排序后 的第 k 小元素,而不是第 k不同 的元素。

你必须找到一个内存复杂度优于 O(n2) 的解决方案。

示例 1:

复制代码
输入:matrix = [[1,5,9],[10,11,13],[12,13,15]], k = 8
输出:13
解释:矩阵中的元素为 [1,5,9,10,11,12,13,13,15],第 8 小元素是 13

示例 2:

复制代码
输入:matrix = [[-5]], k = 1
输出:-5

思路

由题目给出的性质可知,这个矩阵内的元素是从左上到右下递增的。

我们知道整个二维数组中 matrix00 为最小值,matrixn−1n−1 为最大值,现在我们将其分别记作 l 和 r。

可以发现一个性质:任取一个数 mid 满足 l≤mid≤r,那么矩阵中不大于 mid 的数,肯定全部分布在矩阵的左上角。

例如下图,取 mid=8:

我们可以看到,矩阵中大于 mid 的数就和不大于 mid 的数分别形成了两个板块,沿着一条锯齿线将这个矩形分开。其中左上角板块的大小即为矩阵中不大于 mid 的数的数量。

可以这样描述走法:

初始位置在 matrixn−10(即左下角);

设当前位置为 matrixij。若 matrixij≤mid,则将当前所在列的不大于 mid 的数的数量(即 i+1)累加到答案中,并向右移动,否则向上移动;

不断移动直到走出格子为止。

简单说一下关于 "为什么left一定在矩阵中" 的理解 ?

  1. 循环过程中,矩阵中第k小的那个数始终在区间left, right

  2. 循环过程中,left和right可能不是矩阵中的元素,但是left, right中某个元素 "在矩阵中" 且 "满足第k小"

  3. 当right == left时,right, left中只有一个元素了(即left),所以此时left必然 "在矩阵中" 且 "满足第k小"

还有一个陷阱,和搜索旋转排序数组 类似,必须写成 right=mid,left=mid+1,而不能

right=mid-1,left=mid,因为后者在只有两个数的时候可能会跳不出循环。

python 复制代码
from typing import List


class Solution:
    def kthSmallest(self, matrix: List[List[int]], k: int) -> int:
        n=len(matrix)

        def check(mid:int)->bool:
            i,j=n-1,0
            num=0
            while i>=0 and j<n:
                if matrix[i][j]<=mid:
                    num+=(i+1)
                    j+=1
                else:
                    i-=1
            return num>=k


        left, right =matrix[0][0], matrix[-1][-1]
        while left<right:
            mid = (left + right) // 2
            if check(mid):
                right = mid
            else:
                left = mid + 1
        return left

自己的写法:

python 复制代码
import heapq
from typing import List


class Solution:
    def kthSmallest(self, matrix: List[List[int]], k: int) -> int:
        n=len(matrix)
        shuzu=[]
        for i in range(n):
            shuzu.append((matrix[0][i],i))
        heapq.heapify(shuzu)
        while k:
            cur=heapq.heappop(shuzu)
            x,y=cur[1]//n,cur[1]%n
            if x+1<n:
                heapq.heappush(shuzu,(matrix[x+1][y],cur[1]+n))
            k-=1

        return cur[0]
相关推荐
dingzd9520 分钟前
跨境社媒运营越到后面 越比拼账号的表达稳定性
大数据·人工智能·矩阵·内容营销
手写码匠24 分钟前
从零实现 Prompt 工程引擎:结构化提示、自动优化与多轮自省体系
人工智能·深度学习·算法·aigc
无限码力1 小时前
阿里算法岗 0530笔试真题 - 多约束条件下的元素匹配统计
算法·阿里笔试真题·阿里机试真题·阿里算法岗笔试
lqqjuly1 小时前
MLA — 多头潜在注意力深度解析
深度学习·神经网络·算法
吴可可1231 小时前
SolidWorks草图转三维DWG技巧
算法
tyung2 小时前
Go 手写 Wait-Free SPSC 无界队列:无 CAS、无锁、泛型节点池
数据结构·后端·go
redaijufeng2 小时前
C++雾中风景7:闭包
c++·算法·风景
Chen_harmony2 小时前
一、数据结构概念和复杂度计算
数据结构
小欣加油2 小时前
leetcode287寻找重复数
数据结构·c++·算法·leetcode
机汇五金_3 小时前
矩阵机箱有哪些常见结构形式?
线性代数·矩阵