LeetCode 378: Kth Smallest Element in a Sorted Matrix Solution in Python – A Step-by-Step Guide

Imagine you’re given a matrix where each row and column is sorted—like a treasure map with numbers in order—and you need to find the kth smallest element among all its values. That’s the challenge of LeetCode 378: Kth Smallest Element in a Sorted Matrix, a medium-level problem that blends heap usage with matrix traversal. Using Python, we’ll explore two solutions: the Best Solution—a min-heap approach for O(k log k) efficiency—and an Alternative Solution—binary search at O(n log n log M) (where M is the range of values). With examples, clear code breakdowns, and a friendly vibe, this guide will help you unearth that kth element. Let’s start digging!

What Is LeetCode 378: Kth Smallest Element in a Sorted Matrix?

Section link icon

LeetCode 378: Kth Smallest Element in a Sorted Matrix provides an n x n matrix where each row and column is sorted in ascending order, and an integer k. Your task is to return the kth smallest element in the matrix (1-based index, not the kth distinct element). It’s a problem that tests your ability to leverage sorted properties for efficient searching!

Problem Statement

  • Input:
    • matrix: n x n list of integers, sorted by row and column.
    • k: Integer representing the kth smallest element to find (1-based).
  • Output: Integer - The kth smallest element.
  • Rules:
    • Matrix is n x n, each row and column sorted ascending.
    • k is 1-based (k=1 means smallest, k=n² means largest).
    • Return the kth smallest value, not distinct.

Constraints

  • 1 <= n <= 300
  • -10⁹ <= matrix[i][j] <= 10⁹
  • 1 <= k <= n²

Examples

  • Input: matrix = [[1,5,9],[10,11,13],[12,13,15]], k = 8
    • Output: 13
      • Elements: [1, 5, 9, 10, 11, 12, 13, 13, 15], 8th smallest = 13.
  • Input: matrix = [[1,2],[1,3]], k = 2
    • Output: 1
      • Elements: [1, 1, 2, 3], 2nd smallest = 1.
  • Input: matrix = [[1]], k = 1
    • Output: 1
      • Single element, 1st = 1.

These show the kth smallest hunt—let’s solve it!

Understanding the Problem: Finding the Kth Smallest

Section link icon

To tackle LeetCode 378 in Python, we need to: 1. Find the kth smallest element in an n x n sorted matrix. 2. Use the row and column sorting to optimize the search. 3. Handle duplicates (kth smallest, not distinct).

A naive approach might flatten and sort, but that’s O(n² log n²). Instead, we’ll use:

  • Best Solution: Min-heap—O(k log k) time, O(k) space—fast and efficient.
  • Alternative Solution: Binary search—O(n log n log M) time, O(1) space—clever but slower for small k.

Let’s optimize with the best solution!

Best Solution: Min-Heap

Section link icon

Why This Is the Best Solution

The min-heap approach runs in O(k log k) time by maintaining a priority queue of the smallest elements, starting at the top-left corner and expanding only as needed. It leverages the sorted properties to avoid processing all n² elements—making it the most efficient for typical k values (often smaller than n²)!

How It Works

Think of it like a treasure hunt:

  • Heap: Min-heap of (value, row, col) tuples, starting with matrix[0][0].
  • Expand: Pop smallest, push next elements in row and column (if not seen).
  • Repeat: k times to get the kth smallest.
  • Track: Use a set to avoid duplicates.
  • Result: The kth popped value.

It’s like picking the next smallest gem from a sorted grid!

Step-by-Step Example

For matrix = [[1,5,9],[10,11,13],[12,13,15]], k = 8:

  • Init: Heap = [(1, 0, 0)], Seen = {(0, 0)}, Count = 0.
  • Step 1: Pop (1, 0, 0), Count = 1, Push (5, 0, 1), (10, 1, 0).
    • Heap = [(5, 0, 1), (10, 1, 0)].
  • Step 2: Pop (5, 0, 1), Count = 2, Push (9, 0, 2), (11, 1, 1).
    • Heap = [(9, 0, 2), (10, 1, 0), (11, 1, 1)].
  • Step 3: Pop (9, 0, 2), Count = 3, Push (13, 1, 2).
    • Heap = [(10, 1, 0), (11, 1, 1), (13, 1, 2)].
  • Step 4: Pop (10, 1, 0), Count = 4, Push (12, 2, 0).
    • Heap = [(11, 1, 1), (12, 2, 0), (13, 1, 2)].
  • Step 5: Pop (11, 1, 1), Count = 5, Push (13, 2, 1).
    • Heap = [(12, 2, 0), (13, 1, 2), (13, 2, 1)].
  • Step 6: Pop (12, 2, 0), Count = 6, Push (13, 2, 1) (seen), skip.
    • Heap = [(13, 1, 2), (13, 2, 1)].
  • Step 7: Pop (13, 1, 2), Count = 7, Push (15, 2, 2).
    • Heap = [(13, 2, 1), (15, 2, 2)].
  • Step 8: Pop (13, 2, 1), Count = 8, Heap = [(15, 2, 2)].
  • Result: 13 (8th smallest).

For nums = [[1,2],[1,3]], k = 2:

  • Init: Heap = [(1, 0, 0)], Count = 0.
  • Step 1: Pop (1, 0, 0), Count = 1, Push (2, 0, 1), (1, 1, 0).
  • Step 2: Pop (1, 1, 0), Count = 2, Push (3, 1, 1).
  • Result: 1 (2nd smallest).

Code with Explanation

Here’s the Python code using a min-heap:

from heapq import heappush, heappop

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

        # Min-heap: (value, row, col)
        heap = [(matrix[0][0], 0, 0)]
        seen = {(0, 0)}
        count = 0

        # Pop k times to get kth smallest
        while heap:
            val, row, col = heappop(heap)
            count += 1
            if count == k:
                return val

            # Add next elements in row and column
            if col + 1 < n and (row, col + 1) not in seen:
                heappush(heap, (matrix[row][col + 1], row, col + 1))
                seen.add((row, col + 1))
            if row + 1 < n and (row + 1, col) not in seen:
                heappush(heap, (matrix[row + 1][col], row + 1, col))
                seen.add((row + 1, col))

        return -1  # Shouldn't reach here

Explanation

  • Lines 6-9: Handle edge cases: k=1 (smallest), k=n² (largest).
  • Lines 11-12: heap: Min-heap of (value, row, col), start with (0,0), seen: Track visited.
  • Lines 14-24: Heap loop:
    • Pop smallest, increment count.
    • If count = k, return value.
    • Push next in row (col+1) and column (row+1) if valid and unseen.
  • Time: O(k log k)—k heap operations, log k for heap size.
  • Space: O(k)—heap and seen set.

This is like a treasure hunt—grab the kth gem fast!

Comparing the Two Solutions

Section link icon
  • Min-Heap (Best):
    • Time: O(k log k)—heap ops.
    • Space: O(k)—heap.
    • Pros: Fast for small k, intuitive.
    • Cons: Space grows with k.
  • Binary Search (Alternative):
    • Time: O(n log n log M)—range search.
    • Space: O(1)—minimal.
    • Pros: Space-efficient, clever.
    • Cons: Slower for small k.

Min-heap wins for typical k values!

Additional Examples and Edge Cases

Section link icon
  • [[1]], k=1: 1.
  • [[1,2],[3,4]], k=4: 4.
  • Large k: Heap adapts, binary search consistent.

Both work, heap is faster for small k.

Complexity Breakdown

Section link icon
  • Min-Heap:
    • Time: O(k log k).
    • Space: O(k).
  • Binary Search:
    • Time: O(n log n log M).
    • Space: O(1).

Heap excels for efficiency with small k.

Key Takeaways

Section link icon
  • Min-Heap: Pick k smallest—fast and smart!
  • Binary Search: Guess the value—clever and lean!
  • Matrix: Sorting helps.
  • Python Tip: Heapq rocks—see [Python Basics](/python/basics).

Final Thoughts: Find That Kth Element

Section link icon

LeetCode 378: Kth Smallest Element in a Sorted Matrix in Python is a matrix search challenge. Min-heap is the efficiency champ for most cases, while binary search offers a space-saving alternative. Want more? Try LeetCode 215: Kth Largest Element or LeetCode 373: Find K Pairs with Smallest Sums. Ready to search? Visit Solve LeetCode 378 on LeetCode and grab that kth element today!