1339. Maximum Product of Splitted Binary Tree
Problem Description
The problem presents us with a binary tree and asks us to perform an operation where we remove one edge from this tree, effectively splitting it into two separate subtrees. Our goal is to find which edge to remove in order to maximize the product of the sums of the resultant two subtrees' node values. Once we have this maximum product, we must return it modulo 10^9 + 7
, as the number can be quite large.
The complexity here is that we're not just trying to find the sums of any two subtrees - we are specifically looking for an edge whose removal results in two subtrees whose sum-product is the greatest possible among all such removable edges. We have to take care to maximize the product before applying the modulo operation.
Itโs important to note the subtlety in the problem statement that you must maximize the product and then perform the modulo operation rather than taking the modulo of the sums during your calculations.
Intuition
The intuition behind solving this problem involves two main steps:
-
Calculate the total sum of all values in the tree. This will help us determine the total value that we're working to divide in the manner that gives the largest possible product.
-
Perform a depth-first search (DFS) on the tree. During this search, we'll consider each subtree's sum and use it to calculate a candidate product - which is the product of the current subtree's sum and the sum of the rest of the tree. This concept is similar to partitioning an array into subarrays and maximizing the product of their sums.
To calculate the candidate product, we consider the current subtree sum t
and take the product with (total sum - t)
, since removing the edge leading to the subtree would leave a remaining subtree whose sum is total sum - t
. This process is applied at each node of the tree since any node's removal could potentially maximize the product.
The nonlocal variables ans
and s
are used to keep track of the maximum product found so far and the total sum of the tree, respectively. As we recursively calculate the sum of subtrees, we update the product if we find a partition that leads to a larger product.
By performing this exhaustive search, where every node is considered as a potential split point, we ensure that we find the configuration that gives us the maximum possible product. Then we apply the modulo as the last step before returning our answer.
The overall approach is quite efficient because we traverse each node once for the total sum calculation and once more during the DFS, resulting in a time complexity that is linear in the number of nodes in the tree.
Learn more about Tree, Depth-First Search and Binary Tree patterns.
Solution Approach
The solution uses the recursive depth-first search (DFS) algorithm to explore the binary tree. The key data structure used here is the binary tree itself, which consists of nodes defined by the class TreeNode
. Each TreeNode
contains a value and pointers to its left and right child nodes.
Here is a step-by-step breakdown of the solution approach:
-
The
sum
function is a recursive helper function that computes the sum of all node values in the tree. When called with the root of the entire tree, it returns the total sums
of all values. It adds the value of the currentroot
node to the sum of values in the left subtree and the right subtree, recursively applied until leaf nodes are reached. -
The
dfs
function is the main recursive function that performs the depth-first traversal of the tree. For each node, it computes a temporary sumt
, which is the sum of the current node's value and the sums of its left and right subtrees. Every call todfs
returns the sum of the subtree rooted at that node. -
The crucial part of the
dfs
function is where it updates the current answerans
with the maximum product found so far. It calculates the product of the current subtree sumt
and the rest of the tree(s - t)
ift
is less thans
, and updatesans
if this product is larger than the current value ofans
. -
The
mod
variable is used to store the modulo value10^9 + 7
. After the entire tree has been traversed and the maximum product is found, thedfs
function simply returns the answerans
modulomod
. -
The overall recursion starts by calling the
dfs
function from the root of the tree after initializingans
to 0 and computing the total sums
using thesum
function.
In terms of algorithms and patterns, the solution employs a tree traversal and divide-and-conquer strategy. By splitting the tree at different points and comparing the product of the sums, we're effectively trying to find an optimal partition of the tree โ a classic divide-and-conquer approach.
In Python syntax, code referencing values like ans
from an outer scope within a nested function requires the use of the nonlocal
keyword to indicate that the variable is not local to the nested function and should be looked up in the nearest enclosing scope where it is defined.
The DFS traversal and sum calculation both have a time complexity of O(n), where n is the number of nodes in the tree. The space complexity is also O(n) due to the recursion stack depth in a skewed tree, which can be as deep as the number of nodes (in the worst case where the tree is essentially a linked list).
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 take a small binary tree as an example to illustrate the solution approach:
1 1 2 / \ 3 2 3 4 / \ 54 5
Here are the steps following the solution approach outlined above:
-
Total Sum Calculation: Invoke the
sum
function to compute the total sums
of the tree. Thesum
of the tree would be 1 (root value) + 2 (left child value) + 4 (left grandchild value) + 5 (right grandchild value) + 3 (right child value) which equals 15. -
Depth-First Search (DFS) Traversal and Maximizing Product:
- Start the DFS from the root.
- At the root (value 1), calculate the sum of its subtree
(1 + sum(left subtree) + sum(right subtree))
which would be 15. Since we're at the root, the other subtree doesn't exist; we are not removing any edge here. Move to the left child. - At Node 2, calculate its subtree sum, which is
2 + 4 + 5 = 11
. The rest of the tree's sum if we split here would be15 - 11 = 4
(the value of the right subtree, Node 3). Calculate the product candidate11 * 4 = 44
. - At Node 4, its sum is just 4 since it's a leaf node. Removing Node 4 creates subtrees with sums
4
and15 - 4 = 11
. The product for these subtrees would be4 * 11 = 44
. But this product does not exceed our current maximum (also 44), so we don't update. - At Node 5, also a leaf node, the sum is just 5. Removing Node 5 creates subtrees with sums
5
and15 - 5 = 10
. The product would be5 * 10 = 50
. This is greater than our current maximum, so we update our answerans
to 50. - At Node 3, its sum is just 3 since it's also a leaf node. Removing Node 3 creates subtrees with sums
3
and15 - 3 = 12
. The product would be3 * 12 = 36
, which is less than our current maximum. We don't update the answer.
-
Applying Modulo Operation: After conducting the DFS, the highest product calculated is 50. We would then take this number and perform a modulo operation with
10^9 + 7
to get the result that should be returned by the algorithm. Since 50 is much less than10^9 + 7
, our final answer remains 50.
The crucial aspect to realize is that the DFS traversal explores each node, calculating the product for all possible splits and always retains the maximum product found so far. Finally, we apply the modulo operation to the maximum product and return this as our answer.
Solution Implementation
1class Solution:
2 def maxProduct(self, root: Optional[TreeNode]) -> int:
3 # Helper function to calculate the sum of all nodes' values in the tree
4 def calculate_tree_sum(node: Optional[TreeNode]) -> int:
5 if node is None:
6 return 0
7 # Sum the node's value and the sum of left and right subtrees
8 return node.val + calculate_tree_sum(node.left) + calculate_tree_sum(node.right)
9
10 # Helper function to find the maximum product of splitting the tree
11 def find_max_product(node: Optional[TreeNode]) -> int:
12 if node is None:
13 return 0
14
15 # Calculate the sum of the current subtree
16 current_sum = node.val + find_max_product(node.left) + find_max_product(node.right)
17
18 # Check if we can maximize the product using the sum of the current subtree
19 if current_sum < total_sum:
20 nonlocal max_product # Allows us to modify the outer scope's max_product variable
21 max_product = max(max_product, current_sum * (total_sum - current_sum))
22
23 return current_sum
24
25 modulo_base = 10**9 + 7
26 total_sum = calculate_tree_sum(root) # Getting the sum of all nodes in the tree
27 max_product = 0
28 find_max_product(root) # Running the DFS function to find the maximum product
29
30 return max_product % modulo_base # Return the result modulo 10**9 + 7
31
1class Solution {
2 // Global variables to store the maximum product and total sum of the tree
3 private long maxProduct;
4 private long totalSum;
5
6 // Computes the maximum product of the sum of two subtrees of any node
7 public int maxProduct(TreeNode root) {
8 final int MODULO = (int) 1e9 + 7; // The modulo value as per the problem
9 totalSum = calculateSum(root); // Calculate the total sum of all nodes in the tree
10 calculateMaxProduct(root); // Calculate the maximum product
11 return (int) (maxProduct % MODULO); // Return the result under modulo
12 }
13
14 // Helper function to calculate the sum of all nodes in the subtree rooted at 'root'
15 private long calculateSum(TreeNode root) {
16 if (root == null) { // Base case
17 return 0;
18 }
19 // Calculate the sum recursively for left, right subtrees and add the node's value
20 return root.val + calculateSum(root.left) + calculateSum(root.right);
21 }
22
23 // Helper function to update the maximum product
24 private long calculateMaxProduct(TreeNode root) {
25 if (root == null) { // Base case
26 return 0;
27 }
28
29 // Recursively calculate the sum of the current subtree
30 long currentSum = root.val + calculateMaxProduct(root.left) + calculateMaxProduct(root.right);
31
32 // If currentSum is less than the total sum, consider the product of the sum of
33 // this subtree and the sum of the rest of the tree and update maxProduct accordingly
34 if (currentSum < totalSum) {
35 maxProduct = Math.max(maxProduct, currentSum * (totalSum - currentSum));
36 }
37
38 // Return the sum of nodes of the current subtree
39 return currentSum;
40 }
41}
42
43/**
44 * Definition for a binary tree node provided by the problem statement.
45 */
46class TreeNode {
47 int val;
48 TreeNode left;
49 TreeNode right;
50
51 TreeNode() {}
52
53 TreeNode(int val) { this.val = val; }
54
55 TreeNode(int val, TreeNode left, TreeNode right) {
56 this.val = val;
57 this.left = left;
58 this.right = right;
59 }
60}
61
1#include <functional>
2
3// Definition for a binary tree node provided in the problem statement.
4struct TreeNode {
5 int val;
6 TreeNode *left;
7 TreeNode *right;
8 TreeNode() : val(0), left(nullptr), right(nullptr) {}
9 TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
10 TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
11};
12
13class Solution {
14public:
15 int maxProduct(TreeNode* root) {
16 // Use long long type to avoid integer overflow.
17 using ll = long long;
18
19 // Initialize answer to 0.
20 ll answer = 0;
21 // Define the modulo as specified in the problem.
22 const int MOD = 1e9 + 7;
23
24 // Recursive lambda function to calculate the sum of values in the tree.
25 function<ll(TreeNode*)> calculateSum = [&](TreeNode* node) -> ll {
26 if (!node) {
27 return 0;
28 }
29 return node->val + calculateSum(node->left) + calculateSum(node->right);
30 };
31
32 // Calculate the total sum of the tree values.
33 ll totalSum = calculateSum(root);
34
35 // Recursive lambda function to find the max product.
36 function<ll(TreeNode*)> findMaxProduct = [&](TreeNode* node) -> ll {
37 if (!node) {
38 return 0;
39 }
40
41 // Calculate subtree sum.
42 ll subtreeSum = node->val + findMaxProduct(node->left) + findMaxProduct(node->right);
43
44 // If subtree sum is less than total sum, check if the product is the maximum.
45 if (subtreeSum < totalSum) {
46 answer = std::max(answer, subtreeSum * (totalSum - subtreeSum));
47 }
48
49 // Return the sum of the subtree rooted at the current node.
50 return subtreeSum;
51 };
52
53 // Start the depth-first-search traversal from the root.
54 findMaxProduct(root);
55
56 // Return the answer modulo by 1e9+7 as required by problem statement.
57 return answer % MOD;
58 }
59};
60
1// Definition for a binary tree node.
2class TreeNode {
3 val: number;
4 left: TreeNode | null;
5 right: TreeNode | null;
6 constructor(val?: number, left?: TreeNode | null, right?: TreeNode | null) {
7 this.val = val === undefined ? 0 : val;
8 this.left = left === undefined ? null : left;
9 this.right = right === undefined ? null : right;
10 }
11}
12
13// Function to calculate the maximum product of splits of a binary tree.
14function maxProduct(root: TreeNode | null): number {
15 // Calculate the total sum of values in the binary tree.
16 const calculateSum = (node: TreeNode | null): number => {
17 if (!node) {
18 return 0;
19 }
20 return node.val + calculateSum(node.left) + calculateSum(node.right);
21 };
22
23 // Store the total sum of the tree.
24 const totalSum = calculateSum(root);
25
26 // Variable to keep track of the maximum product found.
27 let maximumProduct = 0;
28
29 // The modulo to prevent overflow issues, mathematically equivalent to 10^9 + 7.
30 const mod = 1e9 + 7;
31
32 // DFS function to explore the tree and find the maximum product possible.
33 const dfs = (node: TreeNode | null): number => {
34 if (!node) {
35 return 0;
36 }
37
38 // Calculate the subtree sum including the current node.
39 const subtreeSum = node.val + dfs(node.left) + dfs(node.right);
40
41 // If the subtree sum is less than the total sum, update the maximum product.
42 if (subtreeSum < totalSum) {
43 maximumProduct = Math.max(maximumProduct, subtreeSum * (totalSum - subtreeSum));
44 }
45
46 // Return the subtree sum.
47 return subtreeSum;
48 };
49
50 // Start DFS from the root of the tree.
51 dfs(root);
52
53 // Return the maximum product found, modulo the specified value.
54 return maximumProduct % mod;
55}
56
Time and Space Complexity
Time Complexity
The time complexity of the maxProduct
function primarily depends on two recursive functions: sum
and dfs
.
-
The
sum
function computes the sum of values of all the nodes in the binary tree. This function visits each node exactly once, contributing a time complexity ofO(N)
, whereN
is the total number of nodes in the tree. -
The
dfs
function traverses the binary tree in a similar way to thesum
function, visiting each node to calculate the subtree sums and the maximum product at each step. Since it also visits each node exactly once, its time complexity isO(N)
.
Since we need to run both sum
and dfs
once, and both have the same time complexity O(N)
, the overall time complexity of the maxProduct
function is O(N) + O(N)
, which simplifies to O(N)
.
Space Complexity
The space complexity of the maxProduct
function includes the space used by the recursive call stack and any additional space used by variables and data structures within the function.
-
Both
sum
anddfs
are recursive functions that can go as deep as the height of the tree. Therefore, the space complexity due to the recursive call stack isO(H)
, whereH
is the height of the tree. In the worst case of a skewed tree where the tree behaves like a linked list, the heightH
can beN
, leading to a space complexity ofO(N)
. In the best case, for a balanced tree, this would beO(log(N))
. -
The
maxProduct
function uses a few nonlocal variables (ans
,s
, andmod
) that contribute constant space, which does not depend on the size of the input and thus isO(1)
.
When combined, the overall space complexity of the maxProduct
function is dominated by the space used by the recursive call stack, which is O(H)
. In general terms, considering the worst case, this would be O(N)
, and in the best case, it is O(log(N))
.
Learn more about how to find time and space complexity quickly using problem constraints.
A heap is a ...?
Recommended Readings
Everything About Trees A tree is a type of graph data structure composed of nodes and edges Its main properties are It is acyclic doesn't contain any cycles There exists a path from the root to any node Has N 1 edges where N is the number of nodes in the tree and
https algomonster s3 us east 2 amazonaws com cover_photos dfs svg Depth First Search Prereqs Recursion Review problems recursion_intro Trees problems tree_intro With a solid understanding of recursion under our belts we are now ready to tackle one of the most useful techniques in coding interviews Depth First Search DFS
Binary Tree Min Depth Prereq BFS on Tree problems bfs_intro Given a binary tree find the depth of the shallowest leaf node https algomonster s3 us east 2 amazonaws com binary_tree_min_depth png Explanation We can solve this problem with either DFS or BFS With DFS we traverse the whole tree looking for leaf nodes and record and update the minimum depth as we go With BFS though since we search level by level we are guaranteed to find the shallowest leaf node
Got a question?ย Ask the Monster Assistantย anything you don't understand.
Still not clear? ย Submitย the part you don't understand to our editors. Or join ourย Discord and ask the community.