1439. Find the Kth Smallest Sum of a Matrix With Sorted Rows
Problem Description
The given problem presents a challenge where we are dealing with a matrix called mat
with dimensions m x n
, where each row is sorted in a non-decreasing order. The task here is to create arrays by choosing exactly one element from each row of the matrix. The aim is to find the kth
smallest sum that can be obtained from all the possible arrays created in this way. An array's sum is the total of all the elements within that array.
Intuition
To approach this problem, we utilize a step by step strategy that builds upon the smallest possible sums array by array, while looking ahead by only considering the top k
smallest sums at each step to reduce computational complexity.
Here's how we conceptualize our approach:
-
Begin Small: Initialize a list, let's call it
pre
, with a single element0
, which represents the smallest initial sum (with no elements selected). -
Build Up: For each row in the matrix, we combine the values in
pre
with the new values from the current row. This helps us to generate all possible sums with one more element added to the arrays. -
Stay Within Bounds: To maintain performance and not generate an unnecessarily large number of sums, we only keep the top
k
smallest sums at any given step. This is done by sorting the generated sums and slicing the list to retain only the firstk
elements. -
Iterate Through Rows: Repeat step 2 and 3 for each row of the matrix. With each iteration,
pre
grows to contain the smallest sums resulting from picking exactly one element from each of the rows processed so far. -
Final Answer: After processing all the rows, the last element in
pre
will be thekth
smallest sum, becausepre
is always maintained to be sorted with onlyk
elements. Thus,pre[-1]
yields the desired result.
By employing this incremental approach and limiting the consideration to k
elements at each step, we efficiently find the desired sum without having to explore all possible combinations, which would be infeasible for large matrices.
Learn more about Binary Search and Heap (Priority Queue) patterns.
Solution Approach
The solution employs an efficient approach utilizing simple data structures and Python's list comprehension to handle the potentially large search space in a way that is both manageable and scalable to larger datasets.
Here's a step-by-step breakdown of the implementation:
-
Initialization: We begin by creating a list called
pre
, initialized with a single element,0
. This list represents the sum of elements chosen so far, and with no elements having been considered, the sum is simply0
.pre = [0]
-
Iterative Combination: For every row
cur
in the matrixmat
, we use a nested loop through list comprehension to create new sums. This is where we explore every combination of the previously accumulated sums inpre
with the new elements from the current rowcur
.for cur in mat: pre = [a + b for a in pre for b in cur[:k]]
Inner Loop Explanation: For every element
a
inpre
and for every elementb
in the topk
elements of the currentcur
, compute a new suma + b
. The slicingcur[:k]
ensures that we don't consider more elements than necessary, which is crucial for performance optimization. -
Sorting and Trimming: After combining sums from
pre
and the current row, this new list could be quite large, larger thank
. We need to only keep the smallestk
sums. We do this by first sorting the new sums and then slicing the list to retain only the firstk
elements. The sorting operation here not only ensures that we are keeping the smallest sums but also prepares thepre
list for the next iteration.pre = sorted(pre)[:k]
-
Repeat Process: This process is repeated iteratively for each row of the matrix, each time updating the
pre
list with the smallest sums obtained from choosing exactly one element from each row processed up to that point. -
Extracting Result: After the last iteration, the final list
pre
will be sorted and contain thek
smallest sums of all possible arrays. Thekth
smallest value, which is the answer we need to find, will be the last element of this list.return pre[-1]
By utilizing this approach, we effectively minimize the number of possible sums we have to consider at each step, which is essential for handling inputs where the number of combinations can be extraordinarily high. The algorithm leverages the sorted property of the matrix's rows and Python's efficient list operations to deliver a solution that meets the problem's constraints.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's consider a matrix mat
with two rows and three columns, and suppose we want to find the 2nd smallest sum by choosing exactly one element from each row.
Let mat
be:
[ [1, 3, 4], [2, 5, 7] ]
We want the 2nd smallest sum when picking one element from each row, meaning k=2
in this context.
Step 1: Initialization
We initialize pre
with a single element, 0
, representing the empty sum.
pre = [0]
Step 2: Iterative Combination
Starting with the first row [1, 3, 4]
, combine pre
with this row.
# First row pre = [a + b for a in pre for b in [1, 3, 4][:2]] # We only use the first 2 elements for k=2. # pre becomes [0+1, 0+3] = [1, 3]
We now have all possible sums that can be generated by choosing one element from the first row with pre
only keeping the smallest 2 sums.
Step 3: Sorting and Trimming
Sort pre
and keep only the first k
elements.
pre = sorted(pre)[:2]
# Since pre is already [1, 3], no change is made in this case.
Repeat the iterative combination with the next row [2, 5, 7]
.
Step 4: Repeat Process
Take the next row, and combine each element with the sums in pre
while considering only the first k
elements.
# Second row pre = [a + b for a in pre for b in [2, 5, 7][:2]] # Again, take the first 2 elements only for k=2. # pre becomes [1+2, 1+5, 3+2, 3+5] = [3, 6, 5, 8] (combinations of sums)
Again, sort pre
and keep only the first k
elements.
pre = sorted(pre)[:2]
# After sorting, pre becomes [3, 5], so we keep these as the smallest sums.
Step 5: Extracting Result
After processing all rows, the final list pre
contains the k
smallest sums.
# The 2nd smallest sum is hence return pre[-1] # pre[-1] gives 5, which is the 2nd smallest sum possible.
Therefore, by following the approach, we end up with a pre
list that contains [3, 5]
, and the 2nd smallest sum is the last element in this list, which is 5
. This sum comes from the array [1, 4]
, the second smallest array sum by choosing one element from each row of the given mat
.
Solution Implementation
1from typing import List
2
3class Solution:
4 def kthSmallest(self, matrix: List[List[int]], k: int) -> int:
5 # Initialize a list with a single element 0 to start the accumulation.
6 prev_row_sums = [0]
7
8 # Iterate through each row in the matrix.
9 for row in matrix:
10 # Generate new sums by adding each element in the current row to each
11 # previously computed sum. Since we're only interested in the k smallest sums,
12 # we only consider the first k elements in the row to avoid unnecessary computation.
13 # Then, we sort the resulting sums and keep only the k smallest sums to use
14 # in the next iteration or to decide the final result.
15 prev_row_sums = sorted(sum_1 + sum_2 for sum_1 in prev_row_sums for sum_2 in row[:k])[:k]
16
17 # After processing all rows, the k-th smallest sum is the last element
18 # in the list of k smallest sums.
19 return prev_row_sums[-1]
20
1class Solution {
2 public int kthSmallest(int[][] matrix, int k) {
3 // Get the dimensions of the matrix.
4 int rows = matrix.length;
5 int cols = matrix[0].length;
6
7 // Initialize a list to store the previous row's computations.
8 List<Integer> previousRowSums = new ArrayList<>(k);
9
10 // Initialize a list to store the current row's computations.
11 List<Integer> currentRowSums = new ArrayList<>(cols * k);
12
13 // Start with 0 as the only element for an empty prefix sum.
14 previousRowSums.add(0);
15
16 // Iterate through each row of the matrix.
17 for (int[] row : matrix) {
18 // Clear the current sums list for new calculations.
19 currentRowSums.clear();
20
21 // Combine each element from the previous list with each element of the current row.
22 for (int prevSum : previousRowSums) {
23 for (int value : row) {
24 // Add the sum to the current list.
25 currentRowSums.add(prevSum + value);
26 }
27 }
28
29 // Sort the current list to prepare for selecting the smallest k elements.
30 Collections.sort(currentRowSums);
31
32 // Clear the previous sums list to reuse it for the next iteration.
33 previousRowSums.clear();
34
35 // Take the first k elements from the current list, or the entire list if it's smaller than k.
36 for (int i = 0; i < Math.min(k, currentRowSums.size()); ++i) {
37 // Add each of the smallest elements to the previous sums list.
38 previousRowSums.add(currentRowSums.get(i));
39 }
40 }
41
42 // Return the k-th smallest sum.
43 // Subtract 1 from k because of 0-based indexing in Java lists.
44 return previousRowSums.get(k - 1);
45 }
46}
47
1#include <vector>
2#include <algorithm>
3#include <cstring>
4
5using namespace std;
6
7class Solution {
8public:
9 int kthSmallest(vector<vector<int>>& matrix, int k) {
10 // Initialize the 'previous' array with all zeros and max length as 'k'
11 int previous[k];
12 memset(previous, 0, sizeof(previous));
13 // initial size of the combined list is 1 because we start with zero elements
14 int size = 1;
15
16 // initialize the 'current' array with length up to 'matrix[0].size() * k'
17 // since each addition of a row can at most add that many combinations
18 int current[matrix[0].size() * k]; // This could be optimized using dynamic arrays(Dynamic array/vector is recommended)
19
20 // Iterate through each row of the matrix
21 for (auto& row : matrix) {
22 int index = 0; // Index to store the sum of elements in 'current' array
23
24 // Generate sums for all the possible combinations with the current row
25 for (int j = 0; j < size; ++j) {
26 for (int value : row) {
27 current[index++] = previous[j] + value;
28 }
29 }
30
31 // Sort the 'current' array to bring the smallest sums to the front
32 sort(current, current + index);
33
34 // Only keep the 'k' smallest sums in 'previous' array for next iteration
35 size = min(index, k);
36 for (int j = 0; j < size; ++j) {
37 previous[j] = current[j];
38 }
39 }
40
41 // After combining all rows, the k-th smallest sum is at index k-1
42 return previous[k - 1];
43 }
44};
45
1// Defines a function to find the kth smallest element in a sorted matrix
2function kthSmallest(matrix: number[][], k: number): number {
3 // Initialize a preliminary array with zero, which will keep track of the sum permutations
4 let previousRowSums: number[] = [0];
5
6 // Iterate over each row of the matrix
7 for (const currentRow of matrix) {
8 // Initialize an array to store new sums that include elements from the current row
9 const newRowSums: number[] = [];
10
11 // Iterate over each sum in the previous sums array
12 for (const sum of previousRowSums) {
13 // Add the current element to each of the previous sums
14 for (const element of currentRow) {
15 newRowSums.push(sum + element);
16 }
17 }
18
19 // Sort the sums array and keep only the smallest k sums for the next iteration
20 previousRowSums = newRowSums.sort((a, b) => a - b).slice(0, k);
21 }
22
23 // Return the kth smallest element
24 return previousRowSums[k - 1];
25}
26
Time and Space Complexity
Time Complexity
The time complexity of the given code is determined by the operations executed in the nested loops.
- The outer loop iterates over each row of the matrix
mat
, which we can denote asm
, wherem
is the number of rows in the matrix. - The inner loop essentially computes all pairwise sums between elements in
pre
andcur[:k]
. Ifn
represents the number of columns in the matrix, we have a maximum ofk
elements considered incur
. Thus, in the inner loop, the number of sums will be on the order oflen(pre) * k
. Initially,len(pre) = 1
and can grow up tok
after sorting and truncation. - Sorting the list
pre
can takeO(k log k)
time at most since we only keep the smallestk
elements.
The worst-case computational load at each iteration is dominated by the sorting step, which is O(k log k)
. Thus, the total time complexity of the algorithm is O(m * k * log k)
.
Space Complexity
The space complexity of the code is mainly due to the storage of the list pre
, which maintains a length of at most k
elements through the iterations.
- At each step of the outer loop,
pre
is replaced by a new list of length at mostk
. - The generation of pairwise sums (
a + b for a in pre for b in cur[:k]
) creates an intermediary list of size up tolen(pre) * k
, but since it's immediately sorted and truncated to sizek
, the space consumption does not accumulate over iterations.
Hence, the space complexity of the code is O(k)
.
Learn more about how to find time and space complexity quickly using problem constraints.
Which technique can we use to find the middle of a linked list?
Recommended Readings
Binary Search Speedrun For each of the Speedrun questions you will be given a binary search related problem and a corresponding multiple choice question The multiple choice questions are related to the techniques and template s introduced in the binary search section It's recommended that you have gone through at
https algomonster s3 us east 2 amazonaws com cover_photos heap svg Priority Queue and Heap What is the relationship between priority queue and heap Priority Queue is an Abstract Data Type and Heap is the concrete data structure we use to implement a priority queue Priority Queue A priority queue
LeetCode Patterns Your Personal Dijkstra's Algorithm to Landing Your Dream Job The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we
Want a Structured Path to Master System Design Too? Don’t Miss This!