Given a matrix consisting of only 0,1s, with each of it’s row sorted (i.e all 0s appear before 1s). The problem is to efficiently find the maximum number of 1s present in any row.
For example consider the following matrix.
0 0 1 1
0 0 0 1
0 1 1 1
0 0 0 0
The maximum number of 1s in any row is three which appear in the third row.
The general approach is to find the number of 1s in each row and update the maximum.
There are two ways to find the number of 1s.
- One simple approach is to linearly search for first 1 and find the count of 1s. This will take O(n) time in worst case. So the overall time complexity would be O(n2).
- Another faster method is to search for first 1 using binary search since each row is sorted. This will take only O(log n) time and the overall complexity would be O(n log n).
Even more efficient approach can be as follows.
- Start with the right most element in the first row, keep counting the number of 1s until we reach zero.
- Move to the next row and start checking the elements from previous column
- If the element is zero, just move on to the next row
- Otherwise count the ones until we reach zero or the first column
- Repeat the above step for all the rows. At the end, we get the maximum number 1s.
This approach just runs in O(n+m) where n, m are the number of rows and columns respectively.
Here is the C++ implementation of the above. I have used the lower_bound() method from the standard library to find the left most occurrence of a 1. Look at my previous post to understand how this method works.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include <iostream> | |
#include <vector> | |
#include <algorithm> | |
using namespace std; | |
int getMaximum1( vector< vector<int> > & matrix ) | |
{ | |
int r,c; | |
int result = 0; | |
int nRows = matrix.size(); | |
int nCols = 0; | |
if( nRows > 0 ) | |
{ | |
nCols = matrix[0].size(); | |
for( r = 0; r < nRows; r++ ) | |
{ | |
vector<int>::iterator lb = lower_bound(matrix[r].begin(), matrix[r].end(), 1); | |
if( lb != matrix[r].end() ) //update maximum number of 1s if 1 is found | |
{ | |
result = max(result, matrix[r].end() - lb); | |
} | |
} | |
} | |
return result; | |
} | |
int getMaximum1_fast(vector< vector<int> > & matrix) | |
{ | |
int r,c; | |
int result = 0; | |
int nRows = matrix.size(); | |
int nCols = 0; | |
if( nRows > 0 ) | |
{ | |
nCols = matrix[0].size(); | |
c = nCols-1; | |
r = 0; | |
while( r < nRows ) | |
{ | |
while( c >= 0 && matrix[r][c] == 1 ) | |
{ | |
c--; | |
result++; | |
} | |
if( c < 0 ) //Maximum (i.e no of columns) is reached; we can break | |
break; | |
r++; | |
} | |
} | |
return result; | |
} | |
void Test() | |
{ | |
int mat[4][4] = { {0,0,1,1}, {0,1,1,1}, {1,1,1,1}, {0,0,0,1} }; | |
vector< vector<int> > m(4, vector<int>(4)); | |
for( int i = 0; i < 4; i++ ) | |
{ | |
for( int j = 0; j < 4; j++ ) | |
{ | |
m[i][j] = mat[i][j]; | |
} | |
} | |
cout << getMaximum1(m) << endl; | |
} | |
int main() | |
{ | |
Test(); | |
return 0; | |
} |