forked from kamyu104/GoogleKickStart-2022
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhamiltonian_tour.py3
76 lines (70 loc) · 2.24 KB
/
hamiltonian_tour.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
# Copyright (c) 2022 kamyu. All rights reserved.
#
# Google Kick Start 2022 Round B - Problem D. Hamiltonian Tour
# https://codingcompetitions.withgoogle.com/kickstart/round/00000000008caa74/0000000000acf318
#
# Time: O(R * C)
# Space: O(R * C)
#
def direction(a, b, n):
r, c = divmod(a, n)
nr, nc = divmod(b, n)
if nr-r == 1: return 'S'
if nc-c == 1: return 'E'
if nr-r == -1: return 'N'
if nc-c == -1: return 'W'
def create_node(n, curr, nxts):
r, c = curr
r, c = 2*r, 2*c
if nxts[r*n+c] != -1:
return False
for dr, dc in DIRECTIONS:
nxts[r*n+c] = (r+dr)*n+(c+dc)
r, c = r+dr, c+dc
return True
def merge_node(n, curr, parent, nxts):
if curr < parent:
curr, parent = parent, curr
r, c = curr
pr, pc = parent
if r-pr:
nxts[(2*r)*n+(2*c)] = (2*pr+1)*n+(2*c)
nxts[(2*pr+1)*n+(2*pc+1)] = (2*r)*n+(2*c+1)
else:
nxts[(2*r+1)*n+(2*c)] = (2*pr+1)*n+(2*pc+1)
nxts[(2*pr)*n+(2*pc+1)] = (2*r)*n+(2*c)
def iter_dfs(R, C, B):
nxts = [-1]*((2*R)*(2*C))
create_node(2*C, (0, 0), nxts)
stk = [(1, ((0, 0), None))]
while stk:
step, args = stk.pop()
if step == 1:
curr, parent = args
if parent:
merge_node(2*C, curr, parent, nxts)
stk.append((2, (curr, 0)))
elif step == 2:
curr, i = args
nr, nc = curr[0]+DIRECTIONS[i][0], curr[1]+DIRECTIONS[i][1]
if i+1 < len(DIRECTIONS):
stk.append((2, (curr, i+1)))
if not (0 <= nr < R and 0 <= nc < C and B[nr][nc] == '*' and create_node(2*C, (nr, nc), nxts)):
continue
stk.append((1, ((nr, nc), curr)))
return nxts
def hamiltonian_tour():
R, C = map(int, input().split())
B = [input() for _ in range(R)]
nxts = iter_dfs(R, C, B)
if nxts.count(-1) != 4*sum(row.count('#') for row in B):
return "IMPOSSIBLE"
result = []
curr = 0
while not (result and curr == 0):
result.append(direction(curr, nxts[curr], 2*C))
curr = nxts[curr]
return "".join(result)
DIRECTIONS = [(0, 1), (1, 0), (0, -1), (-1, 0)]
for case in range(int(input())):
print('Case #%d: %s' % (case+1, hamiltonian_tour()))