-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathblockchain.py
130 lines (113 loc) Β· 4.11 KB
/
blockchain.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
# Copyright (c) 2021 kamyu. All rights reserved.
#
# Facebook Hacker Cup 2021 Round 1 - Problem C. Blockchain
# https://www.facebook.com/codingcompetitions/hacker-cup/2021/round-1/problems/C
#
# Time: O(N * MAX_C)
# Space: O(N * MAX_C)
#
from collections import defaultdict
from functools import partial
class UnionFind(object): # Time: O(n * alpha(n)), Space: O(n)
def __init__(self, n):
self.set = range(n)
self.rank = [0]*n
self.size = [1]*n # modified
def find_set(self, x):
stk = []
while self.set[x] != x: # path compression
stk.append(x)
x = self.set[x]
while stk:
self.set[stk.pop()] = x
return x
def union_set(self, x, y):
x_root, y_root = map(self.find_set, (x, y))
if x_root == y_root:
return False
if self.rank[x_root] < self.rank[y_root]: # union by rank
self.set[x_root] = y_root
self.size[y_root] += self.size[x_root] # modified
elif self.rank[x_root] > self.rank[y_root]:
self.set[y_root] = x_root
self.size[x_root] += self.size[y_root] # modified
else:
self.set[y_root] = x_root
self.size[x_root] += self.size[y_root] # modified
self.rank[x_root] += 1
return True
def size_set(self, x): # modified
return self.size[self.find_set(x)]
def addmod(a, b):
return (a+b)%MOD
def submod(a, b):
return (a-b)%MOD
def mulmod(a, b):
return (a*b)%MOD
def iter_postorder_traversal(adj, max_c): # Time: O(N)
def divide(parent, i):
stk.append(partial(conquer, parent, i))
for child, _ in reversed(adj[i]):
if child == parent:
continue
stk.append(partial(divide, i, child))
def conquer(parent, i):
for child, c in adj[i]:
if child == parent:
continue
for j in xrange(c):
dp1[i][j] = addmod(dp1[i][j], dp1[child][j])
stk, dp1 = [], [[1]*max_c for _ in xrange(len(adj))] # dp1[i][j] = number of nodes reachable down from i (incldue i) via capacity >= j+1 edges
stk.append(partial(divide, -1, 0))
while stk:
stk.pop()()
return dp1
def iter_preorder_traversal(adj, total, dp1): # Time: O(N)
def divide(parent, i):
for child, _ in reversed(adj[i]):
if child == parent:
continue
stk.append(partial(divide, i, child))
stk.append(partial(init, parent, i))
def init(parent, i):
for child, c in adj[i]:
if child == parent:
continue
for j in xrange(len(dp2[i])):
if j+1 <= c:
dp2[child][j] = addmod(dp2[i][j], submod(dp1[i][j], dp1[child][j]))
total_i_child, prev = total, 0
for j in reversed(xrange(c)):
curr = mulmod(dp1[child][j], dp2[child][j])
total_i_child = submod(total_i_child, mulmod(j+1, submod(curr, prev)))
prev = curr
result[0] = mulmod(result[0], total_i_child)
result = [1]
stk, dp2 = [], [[0]*len(dp1[0]) for _ in xrange(len(adj))] # dp2[i][j] = number of nodes reachable up from i (exclude i) via capacity >= j+1 edges
stk.append(partial(divide, -1, 0))
while stk:
stk.pop()()
return result[0]
def blockchain():
N = input()
adj = defaultdict(list)
edges = defaultdict(list)
for _ in xrange(N-1):
A, B, C = map(int, raw_input().strip().split())
adj[A-1].append((B-1, C))
adj[B-1].append((A-1, C))
edges[C].append((A-1, B-1))
max_c = max(c for c in edges.iterkeys())
total = 0
uf = UnionFind(N)
for j in reversed(xrange(max_c)):
if j+1 not in edges:
continue
for a, b in edges[j+1]:
total = addmod(total, mulmod(j+1, mulmod(uf.size_set(a), uf.size_set(b))))
uf.union_set(a, b)
dp1 = iter_postorder_traversal(adj, max_c)
return iter_preorder_traversal(adj, total, dp1)
MOD = 10**9+7
for case in xrange(input()):
print 'Case #%d: %s' % (case+1, blockchain())