Skip to content

Commit 7b2eca0

Browse files
alexpantyukhingithub-actionspre-commit-ci[bot]
authored
add distribute coins (TheAlgorithms#7975)
* add distribute coins * updating DIRECTORY.md * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci * fix review notes * fix typehint * fix type in TreeNode Co-authored-by: github-actions <${GITHUB_ACTOR}@users.noreply.github.com> Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
1 parent 5c92b73 commit 7b2eca0

File tree

2 files changed

+136
-0
lines changed

2 files changed

+136
-0
lines changed

DIRECTORY.md

+1
Original file line numberDiff line numberDiff line change
@@ -173,6 +173,7 @@
173173
* [Binary Tree Path Sum](data_structures/binary_tree/binary_tree_path_sum.py)
174174
* [Binary Tree Traversals](data_structures/binary_tree/binary_tree_traversals.py)
175175
* [Diff Views Of Binary Tree](data_structures/binary_tree/diff_views_of_binary_tree.py)
176+
* [Distribute Coins](data_structures/binary_tree/distribute_coins.py)
176177
* [Fenwick Tree](data_structures/binary_tree/fenwick_tree.py)
177178
* [Inorder Tree Traversal 2022](data_structures/binary_tree/inorder_tree_traversal_2022.py)
178179
* [Is Bst](data_structures/binary_tree/is_bst.py)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,135 @@
1+
"""
2+
Author : Alexander Pantyukhin
3+
Date : November 7, 2022
4+
5+
Task:
6+
You are given a tree root of a binary tree with n nodes, where each node has
7+
node.data coins. There are exactly n coins in whole tree.
8+
9+
In one move, we may choose two adjacent nodes and move one coin from one node
10+
to another. A move may be from parent to child, or from child to parent.
11+
12+
Return the minimum number of moves required to make every node have exactly one coin.
13+
14+
Example 1:
15+
16+
3
17+
/ \
18+
0 0
19+
20+
Result: 2
21+
22+
Example 2:
23+
24+
0
25+
/ \
26+
3 0
27+
28+
Result 3
29+
30+
leetcode: https://leetcode.com/problems/distribute-coins-in-binary-tree/
31+
32+
Implementation notes:
33+
User depth-first search approach.
34+
35+
Let n is the number of nodes in tree
36+
Runtime: O(n)
37+
Space: O(1)
38+
"""
39+
40+
from __future__ import annotations
41+
42+
from collections import namedtuple
43+
from dataclasses import dataclass
44+
45+
46+
@dataclass
47+
class TreeNode:
48+
data: int
49+
left: TreeNode | None = None
50+
right: TreeNode | None = None
51+
52+
53+
CoinsDistribResult = namedtuple("CoinsDistribResult", "moves excess")
54+
55+
56+
def distribute_coins(root: TreeNode | None) -> int:
57+
"""
58+
>>> distribute_coins(TreeNode(3, TreeNode(0), TreeNode(0)))
59+
2
60+
>>> distribute_coins(TreeNode(0, TreeNode(3), TreeNode(0)))
61+
3
62+
>>> distribute_coins(TreeNode(0, TreeNode(0), TreeNode(3)))
63+
3
64+
>>> distribute_coins(None)
65+
0
66+
>>> distribute_coins(TreeNode(0, TreeNode(0), TreeNode(0)))
67+
Traceback (most recent call last):
68+
...
69+
ValueError: The nodes number should be same as the number of coins
70+
>>> distribute_coins(TreeNode(0, TreeNode(1), TreeNode(1)))
71+
Traceback (most recent call last):
72+
...
73+
ValueError: The nodes number should be same as the number of coins
74+
"""
75+
76+
if root is None:
77+
return 0
78+
79+
# Validation
80+
def count_nodes(node: TreeNode | None) -> int:
81+
"""
82+
>>> count_nodes(None):
83+
0
84+
"""
85+
if node is None:
86+
return 0
87+
88+
return count_nodes(node.left) + count_nodes(node.right) + 1
89+
90+
def count_coins(node: TreeNode | None) -> int:
91+
"""
92+
>>> count_coins(None):
93+
0
94+
"""
95+
if node is None:
96+
return 0
97+
98+
return count_coins(node.left) + count_coins(node.right) + node.data
99+
100+
if count_nodes(root) != count_coins(root):
101+
raise ValueError("The nodes number should be same as the number of coins")
102+
103+
# Main calculation
104+
def get_distrib(node: TreeNode | None) -> CoinsDistribResult:
105+
"""
106+
>>> get_distrib(None)
107+
namedtuple("CoinsDistribResult", "0 2")
108+
"""
109+
110+
if node is None:
111+
return CoinsDistribResult(0, 1)
112+
113+
left_distrib_moves, left_distrib_excess = get_distrib(node.left)
114+
right_distrib_moves, right_distrib_excess = get_distrib(node.right)
115+
116+
coins_to_left = 1 - left_distrib_excess
117+
coins_to_right = 1 - right_distrib_excess
118+
119+
result_moves = (
120+
left_distrib_moves
121+
+ right_distrib_moves
122+
+ abs(coins_to_left)
123+
+ abs(coins_to_right)
124+
)
125+
result_excess = node.data - coins_to_left - coins_to_right
126+
127+
return CoinsDistribResult(result_moves, result_excess)
128+
129+
return get_distrib(root)[0]
130+
131+
132+
if __name__ == "__main__":
133+
import doctest
134+
135+
doctest.testmod()

0 commit comments

Comments
 (0)