-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathwiki_race.py3
82 lines (78 loc) · 2.48 KB
/
wiki_race.py3
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
# Copyright (c) 2023 kamyu. All rights reserved.
#
# Meta Hacker Cup 2023 Round 2 - Problem C. Wiki Race
# https://www.facebook.com/codingcompetitions/hacker-cup/2023/round-2/problems/C
#
# Time: O(N + SUM_LEN_S), SUM_LEN_S = sum(len(x) for i in range(N) for x in S[i])
# Space: O(N + SUM_LEN_S)
#
def wiki_race():
def iter_dfs():
cnt = [0]*len(lookup)
leaves = 0
stk = [0]
while stk:
u = stk.pop()
while len(adj[u]) == 1:
v = adj[u][0]
for x in S[v]:
S[u].add(x)
adj[u] = adj[v]
for x in S[u]:
cnt[x] += 1
if not adj[u]:
leaves += 1
continue
for v in adj[u]:
stk.append(v)
return cnt, leaves
def iter_dfs2(x):
ret = [0]
stk = [(1, (0, ret))]
while stk:
step, args = stk.pop()
if step == 1:
u, ret = args
cnt = [0]
stk.append((4, (u, cnt, ret)))
stk.append((2, (u, 0, cnt)))
elif step == 2:
u, i, cnt = args
if i == len(adj[u]):
continue
ret = [0]
stk.append((2, (u, i+1, cnt)))
stk.append((3, (ret, cnt)))
stk.append((1, (adj[u][i], ret)))
elif step == 3:
ret, cnt = args
cnt[0] += ret[0]
elif step == 4:
u, cnt, ret = args
if not adj[u]:
ret[0] = int(x in S[u])
continue
if cnt[0] == len(adj[u]):
ret[0] = 1
continue
if cnt[0]+1 == len(adj[u]):
ret[0] = int(x in S[u])
continue
return 0
return int(ret[0] == 1)
N = int(input())
P = list(map(lambda x: int(x)-1, input().split()))
S = [input().split()[1:] for _ in range(N)]
lookup = {}
for i in range(N):
for x in S[i]:
if x not in lookup:
lookup[x] = len(lookup)
S[i] = {lookup[x] for x in S[i]}
adj = [[] for _ in range(N)]
for i, p in enumerate(P, 1):
adj[p].append(i)
cnt, leaves = iter_dfs()
return sum(iter_dfs2(k) for k, v in enumerate(cnt) if v >= leaves)
for case in range(int(input())):
print('Case #%d: %s' % (case+1, wiki_race()))