image.png

思考

  1. class Solution {
  2. public int kthSmallest(int[][] matrix, int k) {
  3. if(matrix.length*matrix[0].length<k)
  4. return -1;
  5. PriorityQueue<Integer> heap = new PriorityQueue<>();
  6. for(int i=0;i<matrix.length;i++)
  7. for(int j=0;j<matrix[0].length;j++)
  8. heap.add(matrix[i][j]);
  9. for(int i=0;i<k-1;i++)
  10. heap.poll();
  11. return heap.poll();
  12. }
  13. }

二分查找

image.png

 public int kthSmallest(int[][] matrix, int k) {
        int row = matrix.length;
        int col = matrix[0].length;
        int left = matrix[0][0];
        int right = matrix[row - 1][col - 1];
        while (left < right) {
            // 每次循环都保证第K小的数在start~end之间,当start==end,第k小的数就是start
            int mid = (left + right) / 2;
            // 找二维矩阵中<=mid的元素总个数
            int count = findNotBiggerThanMid(matrix, mid, row, col);
            if (count < k) {
                // 第k小的数在右半部分,且不包含mid
                left = mid + 1;
            } else {
                // 第k小的数在左半部分,可能包含mid
                right = mid;
            }
        }
        return right;
 }

private int findNotBiggerThanMid(int[][] matrix, int mid, int row, int col) {
        // 以列为单位找,找到每一列最后一个<=mid的数即知道每一列有多少个数<=mid
        int i = row - 1;
        int j = 0;
        int count = 0;
        while (i >= 0 && j < col) {
            if (matrix[i][j] <= mid) {
                // 第j列有i+1个元素<=mid
                count += i + 1;
                j++;
            } else {
                // 第j列目前的数大于mid,需要继续在当前列往上找
                i--;
            }
        }
        return count;
    }