Problem Statement:
Given an m x n
matrix matrix
and a target value target
, write a function to search for target
in matrix
. The matrix has the following properties:
- Integers in each row are sorted from left to right.
- The first integer of each row is greater than the last integer of the previous row.
You need to implement the function:
bool searchMatrix(vector<vector<int>>& matrix, int target);
Optimal Approach:
The most efficient way to solve this problem is by treating the 2D matrix as a 1D sorted array. This approach can be solved using Binary Search in O(log(m * n)) time, where m
is the number of rows and n
is the number of columns.
Steps:
- Start by performing a binary search on the rows.
- Once the correct row is found, perform a binary search on the elements within that row.
LeetCode-Compatible Code Implementation:
class Solution {
public:
bool searchMatrix(vector<vector<int>>& matrix, int target) {
int m = matrix.size();
int n = matrix[0].size();
int low = 0, high = m * n - 1;
while (low <= high) {
int mid = low + (high - low) / 2;
int midValue = matrix[mid / n][mid % n];
if (midValue == target) {
return true;
} else if (midValue < target) {
low = mid + 1;
} else {
high = mid - 1;
}
}
return false;
}
};
Step-by-Step Explanation of Code:
- Initialize variables:
m
is the number of rows andn
is the number of columns. - Set up the binary search range:
low
starts at 0 andhigh
starts atm * n - 1
(the last element in the flattened array). - Binary Search loop:
- Calculate the middle index
mid
. - Convert
mid
into a 2D matrix index by dividing byn
for the row and taking the remainder for the column. - Compare the value at
matrix[mid / n][mid % n]
with the target.
- Calculate the middle index
- Adjust the binary search bounds: If the target is less than the current value, move the
high
pointer tomid - 1
; if greater, move thelow
pointer tomid + 1
. - Return the result: If the target is found, return
true
. Otherwise, returnfalse
.
Dry Run / Execution Steps:
Let’s dry run the algorithm with a sample input.
Input:
matrix = [
[1, 4, 7, 11],
[2, 5, 8, 12],
[3, 6, 9, 16],
[10, 13, 14, 17]
]
target = 5
Step-by-Step Execution:
- Initialize
low = 0
,high = 15
(becausem * n = 4 * 4 = 16
). - Compute
mid = (0 + 15) / 2 = 7
.matrix[7 / 4][7 % 4] = matrix[1][3] = 12
.- Since 12 is greater than 5, update
high = 6
.
- Compute
mid = (0 + 6) / 2 = 3
.matrix[3 / 4][3 % 4] = matrix[0][3] = 11
.- Since 11 is greater than 5, update
high = 2
.
- Compute
mid = (0 + 2) / 2 = 1
.matrix[1 / 4][1 % 4] = matrix[0][1] = 4
.- Since 4 is less than 5, update
low = 2
.
- Compute
mid = (2 + 2) / 2 = 2
.matrix[2 / 4][2 % 4] = matrix[0][2] = 7
.- Since 7 is greater than 5, update
high = 1
.
- At this point, the binary search terminates.
Output:
The output will be false
.
Alternative Approaches & Why Not?
- Brute Force: Iterate over each element in the matrix. This would take O(m * n) time, which is inefficient.
- Sorting: You could flatten the matrix and sort it, but this would take O(m * n log(m * n)) time, which is less optimal.
- Dynamic Programming: Not necessary for this problem, as a simple binary search is enough.
- Two Pointers: A two-pointer approach might work, but binary search is more efficient.
Best Solution & Why It’s Best:
The binary search approach is the most efficient, achieving O(log(m * n)) time complexity and O(1) space complexity. It's optimal because we leverage the sorted nature of the matrix to reduce the search space exponentially.
Approach | Time Complexity | Space Complexity |
---|---|---|
Brute Force | O(m * n) | O(1) |
Sorting | O(m * n log(m * n)) | O(m * n) |
Binary Search | O(log(m * n)) | O(1) |
Complexity Analysis:
-
Time Complexity:
- Brute Force: O(m * n)
- Sorting: O(m * n log(m * n))
- Binary Search: O(log(m * n))
-
Space Complexity:
- Brute Force: O(1)
- Sorting: O(m * n)
- Binary Search: O(1)
Conclusion:
The binary search method is the best approach for this problem due to its optimal time complexity and space efficiency. Similar problems that might help practice this approach include: