1975. Maximum Matrix Sum
Problem Description
Given an n x n
matrix filled with integers, the task is to perform strategic operations to maximize the sum of all elements in the matrix. An operation consists of selecting any two adjacent elements (elements that share a border) and multiplying both by -1
. This toggling between positive and negative can be done as many times as desired. The goal is to determine the highest possible sum of all elements in the matrix after performing zero or more of these operations.
Intuition
The key to solving this problem is realizing that each operation can be used to negate the effect of negative numbers in the matrix. It's generally advantageous to have all numbers positive for the highest sum. However, since operations are limited to pairs of adjacent elements, it may not always be possible to convert all negatives to positives.
The strategy is as follows:
-
Calculate the total sum of all the absolute values of the elements in the matrix. This is the maximum sum the matrix can possibly have if we could individually toggle each element to be positive.
-
Track the smallest absolute value of the elements in the matrix. This value is important because if there's an odd number of negative elements that can't be paired, the smallest element's negativity will have the smallest negative impact on the total sum.
-
Count the number of negative elements in the matrix. If this count is even, it means every negative element can be paired with another negative to turn both into positives through one operation.
-
If the count of negative elements is odd, there will remain one unpaired negative element affecting the total sum. In this case, subtract twice the smallest absolute value found in the matrix from the total sum to account for the impact of the remaining negative number after all possible pairs have been negated. If the smallest value is zero, it means an operation can be done without impacting the total sum, as multiplying zero by
-1
does not change the value.
The solution leverages these intuitions to determine the maximum sum.
Learn more about Greedy patterns.
Solution Approach
To implement the solution, we perform a series of steps that follow the intuition described earlier. Here's a breakdown of the algorithm based on the code provided:
-
Initialize a sum variable
s
to 0, which will store the total sum of absolute values of the matrix elements. Also, create a countercnt
to keep track of the number of negative elements and a variablemi
to keep the minimum absolute value observed in the matrix. -
Iterate through each element of the matrix using a nested loop (iterating first through rows, then through elements within each row). For every element
v
in the matrix:- Add the absolute value of the element to
s
(s += abs(v)
), progressively building the total sum of absolute values. - Update the minimum absolute value
mi
if the absolute value of the current element is less than the currentmi
(mi = min(mi, abs(v))
). - If the current element
v
is negative, increment the countercnt
(if v < 0: cnt += 1
). This helps keep track of how many negative numbers are in the matrix, crucial for deciding the following steps.
- Add the absolute value of the element to
-
After iterating through the matrix:
-
Check if the number of negative elements
cnt
is even or the smallest valuemi
is zero:- If
cnt
is even, it's possible to negate all negative elements by using the operation, and the highest sum can be obtained by simply summing all absolute values. Return the sums
. - If
mi
is zero, it means one of the elements is zero, and no subtraction is needed as multiplying zero by-1
multiple times will not change the sum. So, returns
.
- If
-
If the number of negative elements
cnt
is odd andmi
is not zero, it means that not all negative elements can be negated, and there will be one negative element affecting the total sum. Returns - mi * 2
, subtracting twice the smallest absolute value to account for the remaining negative element's impact on the total sum.
-
This approach uses simple data structures (just integers and loops over the 2D matrix), with the main pattern being the calculation of sums, tracking the smallest value, and counting occurrences of a particular condition (negativity in this case), which are fairly common operations in matrix manipulation problems.
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 small 3x3
matrix as an example:
[ [ -1, -2, 3], [ 4, 5, -6], [ -7, 8, 9] ]
Following the solution approach:
-
Initialize
s
as 0,cnt
as 0, andmi
as infinity (or a very large number). -
Iterate through the matrix and calculate:
s
(the sum of absolute values), which is| -1 | + | -2 | + | 3 | + | 4 | + | 5 | + | -6 | + | -7 | + | 8 | + | 9 |
=1 + 2 + 3 + 4 + 5 + 6 + 7 + 8 + 9
=45
.mi
(the smallest absolute value), which ismin(infinity, 1, 2, 3, 4, 5, 6, 7, 8, 9)
=1
.cnt
(the count of negative numbers), there are4
negatives:-1, -2, -6, -7
socnt = 4
.
-
After the iteration, check the conditions:
cnt
is4
, which is even. Therefore, we don't need to subtract anything froms
, because we can pair all negatives and multiply by-1
to turn them positive.- Since
cnt
is even, we can returns
which equals45
as our answer.
According to the strategy, the operations performed would be:
- Multiplying
-1
and-2
by-1
to turn them positive. - Multiplying
-6
and-7
by-1
to turn them positive.
Therefore, the highest possible sum after applying the operations is 45
.
Solution Implementation
1class Solution:
2 def maxMatrixSum(self, matrix: List[List[int]]) -> int:
3 # Initialize total sum, negative count, and minimum positive value
4 total_sum = negative_count = 0
5 minimum_positive_value = float('inf') # Represents infinity
6
7 # Iterate over each row in the matrix
8 for row in matrix:
9 # Iterate over each value in the row
10 for value in row:
11 # Add the absolute value of the current element to the total sum
12 total_sum += abs(value)
13
14 # Record the minimum positive value (smallest absolute value)
15 minimum_positive_value = min(minimum_positive_value, abs(value))
16
17 # If the current value is negative, increment the negative count
18 if value < 0:
19 negative_count += 1
20
21 # If there is an even number of negatives (or a zero element, which can flip sign without penalty),
22 # the result is simply the total sum of absolute values
23 if negative_count % 2 == 0 or minimum_positive_value == 0:
24 return total_sum
25
26 # If there's an odd number of negatives, subtract twice the minimum positive
27 # value to account for the one value that will remain negative
28 return total_sum - minimum_positive_value * 2
29
1class Solution {
2 public long maxMatrixSum(int[][] matrix) {
3 long sum = 0; // Initialize a sum variable to hold the total sum of matrix elements
4 int negativeCount = 0; // Counter for the number of negative elements in the matrix
5 int minAbsValue = Integer.MAX_VALUE; // Initialize to the maximum possible value to track the smallest absolute value seen
6
7 // Loop through each row of the matrix
8 for (int[] row : matrix) {
9 // Loop through each value in the row
10 for (int value : row) {
11 sum += Math.abs(value); // Add the absolute value of the element to the sum
12 // Find the smallest absolute value in the matrix
13 minAbsValue = Math.min(minAbsValue, Math.abs(value));
14 // If the element is negative, increment the negativeCount
15 if (value < 0) {
16 negativeCount++;
17 }
18 }
19 }
20
21 // If the count of negative numbers is even or there's at least one zero, return the sum of absolute values
22 if (negativeCount % 2 == 0 || minAbsValue == 0) {
23 return sum;
24 }
25
26 // Since the negative count is odd, we subtract twice the smallest absolute value to maximize the matrix sum
27 return sum - (minAbsValue * 2);
28 }
29}
30
1class Solution {
2public:
3 long long maxMatrixSum(vector<vector<int>>& matrix) {
4 long long sumOfAbsoluteValues = 0; // This variable will store summation of absolute values of all elements in matrix
5 int negativeCount = 0; // Counter for the number of negative elements in the matrix
6 int minAbsValue = INT_MAX; // This variable will keep track of the smallest absolute value encountered
7
8 // Loop over each row in the matrix
9 for (auto& row : matrix) {
10 // Loop over each element in the row
11 for (int& value : row) {
12 // Add the absolute value of the current element to the total sum
13 sumOfAbsoluteValues += abs(value);
14 // Update the smallest absolute value encountered if current absolute value is smaller
15 minAbsValue = min(minAbsValue, abs(value));
16 // If the current element is negative, increment the negative counter
17 negativeCount += value < 0;
18 }
19 }
20
21 // If the number of negative values is even or the minimum absolute value is 0,
22 // we can make all elements non-negative without decreasing the sum of absolute values.
23 if (negativeCount % 2 == 0 || minAbsValue == 0) {
24 return sumOfAbsoluteValues;
25 }
26
27 // If the number of negative values is odd, we subtract twice the smallest
28 // absolute value to compensate for the one element that will remain negative.
29 return sumOfAbsoluteValues - minAbsValue * 2;
30 }
31};
32
1/**
2 * Calculates the maximum absolute sum of any submatrix of the given matrix.
3 *
4 * @param {number[][]} matrix - The 2D array of numbers representing the matrix.
5 * @return {number} - The maximum absolute sum possible by potentially negating any submatrix element.
6 */
7function maxMatrixSum(matrix: number[][]): number {
8 let negativeCount = 0; // Count of negative numbers in the matrix
9 let sum = 0; // Sum of the absolute values of the elements in the matrix
10 let minAbsValue = Infinity; // Smallest absolute value found in the matrix
11
12 // Iterate through each row of the matrix
13 for (const row of matrix) {
14 // Iterate through each value in the row
15 for (const value of row) {
16 sum += Math.abs(value); // Add the absolute value to the sum
17 minAbsValue = Math.min(minAbsValue, Math.abs(value)); // Update min absolute value if necessary
18 negativeCount += value < 0 ? 1 : 0; // Increment count if the value is negative
19 }
20 }
21
22 // If the count of negative numbers is even, the sum is already maximized
23 if (negativeCount % 2 == 0) {
24 return sum;
25 }
26
27 // Otherwise, subtract double the smallest absolute value to negate an odd count of negatives
28 return sum - minAbsValue * 2;
29}
30
31// Example usage:
32// const matrix = [[1, -1], [-1, 1]];
33// const result = maxMatrixSum(matrix);
34// console.log(result); // Output should be 4
35
Time and Space Complexity
Time Complexity
The given code iterates through all elements of the matrix
with dimension n x n
. For each element, it performs a constant number of operations: calculating the absolute value, updating the sum s
, comparing with the minimum value mi
, and incrementing a counter cnt
if the value is negative.
- Iterating through all elements takes
O(n^2)
time, wheren
is the dimension of the square matrix (since there aren
rows andn
columns). - All the operations inside the nested loops are constant time operations.
Hence, combining these, the overall time complexity of the code is O(n^2)
.
Space Complexity
The space complexity is determined by the additional space required by the code, not including the space taken by the inputs.
- The variables
s
,cnt
, andmi
use constant space. - There are no additional data structures that grow with the size of the input.
Therefore, the space complexity of the code is O(1)
, which indicates constant space usage regardless of the input size.
Learn more about how to find time and space complexity quickly using problem constraints.
Which algorithm should you use to find a node that is close to the root of the tree?
Recommended Readings
Greedy Introduction div class responsive iframe iframe src https www youtube com embed WTslqPbj7I title YouTube video player frameborder 0 allow accelerometer autoplay clipboard write encrypted media gyroscope picture in picture web share allowfullscreen iframe div When do we use greedy Greedy algorithms tend to solve optimization problems Typically they will ask you to calculate the max min of some value Commonly you may see this phrased in the problem as max min longest shortest largest smallest etc These keywords can be identified by just scanning
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
Recursion Recursion is one of the most important concepts in computer science Simply speaking recursion is the process of a function calling itself Using a real life analogy imagine a scenario where you invite your friends to lunch https algomonster s3 us east 2 amazonaws com recursion jpg You first
Want a Structured Path to Master System Design Too? Don’t Miss This!