# Dynamic Programming on Trees | Set-1

Dynamic Programming(DP) is a technique to solve problems by breaking them down into overlapping sub-problems which follows the optimal substructure. There are various problems using DP like subset sum, knapsack, coin change etc. DP can also be applied on trees to solve some specific problems.

**Pre-requisite:** DFS

Given a tree with N nodes and N-1 edges, calculate the maximum sum of the node values from root to any of the leaves without re-visiting any node.

Given above is a diagram of a tree with **N=14** nodes and **N-1=13** edges. The values at node being **3, 2, 1, 10, 1, 3, 9, 1, 5, 3, 4, 5, 9 and 8 respectively for nodes 1, 2, 3, 4….14.**

The diagram below shows all the paths from root to leaves :

All the paths are marked by different colors :

Path 1(red, 3-2-1-4) : sum of all node values = 10

Path 2(orange, 3-2-1-5) : sum of all node values = 11

Path 3(yellow, 3-2-3) : sum of all node values = 8

Path 4(green, 3-1-9-9) : sum of all node values = 22

Path 5(violet, 3-1-9-8) : sum of all node values = 21

Path 6(pink, 3-10-1) : sum of all node values = 14

Path 7(blue, 3-10-5) : sum of all node values = 18

Path 8(brown, 3-10-3) : sum of all node values = 16

The answer is 22, as Path 4 has the maximum sum of values of nodes in its path from a root to leaves.

**The greedy approach fails in this case**. Starting from the root and take 3 from the first level, 10 from the next level and 5 from the third level greedily. Result is path-7 if after following greedy approach, hence do not apply greedy approach over here.

The problem can be solved using **Dynamic Programming on trees. **Start memoizing from the leaves and add the maximum of leaves to the root of every sub-tree. At the last step, there will be root and the sub-tree under it, adding the value at node and maximum of sub-tree will give us the maximum sum of the node values from root to any of the leaves.

The diagram above shows how to **start from the leaves and add the maximum of leaves of a sub-tree to its root**. Move upward and repeat the same procedure of storing the maximum of every sub-tree leaves and adding it to its root. In this example, the maximum of node 11 and 12 is taken to count and then added to node 5(**In this sub-tree, 5 is the root and 11, 12 are its leaves**). Similarly, the maximum of node 13 and 15 is taken to count and then added to node 7. Repeat the steps for every sub-tree till we reach the node.

Let DP_{i} be the maximum summation of node values in the path between i and any of its leaves moving downwards. **Traverse the tree using DFS traversal**. Store the maximum of all the leaves of the sub-tree, and add it to the root of the sub-tree. At the end, DP_{1} will have the maximum sum of the node values from root to any of the leaves without re-visiting any node.

Below is the implementation of the above idea :

## CPP

`// CPP code to find the maximum path sum ` `#include <bits/stdc++.h> ` `using` `namespace` `std; ` ` ` `int` `dp[100]; ` ` ` `// function for dfs traversal and to store the ` `// maximum value in dp[] for every node till the leaves ` `void` `dfs(` `int` `a[], vector<` `int` `> v[], ` `int` `u, ` `int` `parent) ` `{ ` ` ` `// initially dp[u] is always a[u] ` ` ` `dp[u] = a[u - 1]; ` ` ` ` ` `// stores the maximum value from nodes ` ` ` `int` `maximum = 0; ` ` ` ` ` `// traverse the tree ` ` ` `for` `(` `int` `child : v[u]) { ` ` ` ` ` `// if child is parent, then we continue ` ` ` `// without recursing further ` ` ` `if` `(child == parent) ` ` ` `continue` `; ` ` ` ` ` `// call dfs for further traversal ` ` ` `dfs(a, v, child, u); ` ` ` ` ` `// store the maximum of previous visited node ` ` ` `// and present visited node ` ` ` `maximum = max(maximum, dp[child]); ` ` ` `} ` ` ` ` ` `// add the maximum value returned to the parent node ` ` ` `dp[u] += maximum; ` `} ` ` ` `// function that returns the maximum value ` `int` `maximumValue(` `int` `a[], vector<` `int` `> v[]) ` `{ ` ` ` `dfs(a, v, 1, 0); ` ` ` `return` `dp[1]; ` `} ` ` ` `// Driver Code ` `int` `main() ` `{ ` ` ` `// number of nodes ` ` ` `int` `n = 14; ` ` ` ` ` `// adjacency list ` ` ` `vector<` `int` `> v[n + 1]; ` ` ` ` ` `// create undirected edges ` ` ` `// initialize the tree given in the diagram ` ` ` `v[1].push_back(2), v[2].push_back(1); ` ` ` `v[1].push_back(3), v[3].push_back(1); ` ` ` `v[1].push_back(4), v[4].push_back(1); ` ` ` `v[2].push_back(5), v[5].push_back(2); ` ` ` `v[2].push_back(6), v[6].push_back(2); ` ` ` `v[3].push_back(7), v[7].push_back(3); ` ` ` `v[4].push_back(8), v[8].push_back(4); ` ` ` `v[4].push_back(9), v[9].push_back(4); ` ` ` `v[4].push_back(10), v[10].push_back(4); ` ` ` `v[5].push_back(11), v[11].push_back(5); ` ` ` `v[5].push_back(12), v[12].push_back(5); ` ` ` `v[7].push_back(13), v[13].push_back(7); ` ` ` `v[7].push_back(14), v[14].push_back(7); ` ` ` ` ` `// values of node 1, 2, 3....14 ` ` ` `int` `a[] = { 3, 2, 1, 10, 1, 3, 9, 1, 5, 3, 4, 5, 9, 8 }; ` ` ` ` ` `// function call ` ` ` `cout << maximumValue(a, v); ` ` ` ` ` `return` `0; ` `} ` |

*chevron_right*

*filter_none*

**Output:**

22

**Time Complexity : ** O(N), where N is the number of nodes.

## Recommended Posts:

- Dynamic Programming on Trees | Set 2
- Sum over Subsets | Dynamic Programming
- Bitmasking and Dynamic Programming | Set-2 (TSP)
- Double Knapsack | Dynamic Programming
- Number of Unique BST with a given key | Dynamic Programming
- Dynamic Programming vs Divide-and-Conquer
- How to solve a Dynamic Programming Problem ?
- Top 20 Dynamic Programming Interview Questions
- Dynamic Programming | Building Bridges
- Optimal Substructure Property in Dynamic Programming | DP-2
- Overlapping Subproblems Property in Dynamic Programming | DP-1
- Compute nCr % p | Set 1 (Introduction and Dynamic Programming Solution)
- Distinct palindromic sub-strings of the given string using Dynamic Programming
- Travelling Salesman Problem | Set 1 (Naive and Dynamic Programming)
- Understanding The Coin Change Problem With Dynamic Programming

If you like GeeksforGeeks and would like to contribute, you can also write an article using contribute.geeksforgeeks.org or mail your article to contribute@geeksforgeeks.org. See your article appearing on the GeeksforGeeks main page and help other Geeks.

Please Improve this article if you find anything incorrect by clicking on the "Improve Article" button below.