solution.py 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  1. #!/usr/bin/python3
  2. import sys
  3. import time as timepkg
  4. from collections import deque
  5. moves = {'>': (1, 0), 'v': (0, 1), '<': (-1, 0), '^': (0, -1)}
  6. def print_grid(blizzards, positions, width, height, start, goal):
  7. for y in range(height):
  8. for x in range(width):
  9. if (x == 0 or y == 0 or x == width - 1 or y == height - 1) and not ((x, y) == start or (x, y) == goal):
  10. print('#', end='')
  11. elif (x, y) in positions:
  12. print('E', end='')
  13. else:
  14. is_blizzard = [0 for i in range(4)]
  15. is_blizzard[0] = ('>', x, y) in blizzards
  16. is_blizzard[1] = ('v', x, y) in blizzards
  17. is_blizzard[2] = ('<', x, y) in blizzards
  18. is_blizzard[3] = ('^', x, y) in blizzards
  19. if sum(is_blizzard) > 1:
  20. print('+', end='')
  21. elif ('>', x, y) in blizzards:
  22. print('>', end='')
  23. elif ('v', x, y) in blizzards:
  24. print('v', end='')
  25. elif ('<', x, y) in blizzards:
  26. print('<', end='')
  27. elif ('^', x, y) in blizzards:
  28. print('^', end='')
  29. else:
  30. print('.', end='')
  31. print()
  32. print()
  33. def parse_input():
  34. blizzards = set()
  35. width = height = 0
  36. for line in sys.stdin:
  37. stripped = line.strip()
  38. if len(stripped) == 0: continue
  39. [blizzards.add((stripped[x], x, height)) for x in range(len(stripped)) if stripped[x] in ('>', '<', '^', 'v')]
  40. width = max(width, len(stripped))
  41. height += 1
  42. print(blizzards, width, height)
  43. return blizzards, width, height
  44. def simulate_blizzards(blizzards, width, height):
  45. new_blizzards = set()
  46. # move blizzards
  47. for blizzard in blizzards:
  48. direction, x, y = blizzard
  49. #print(direction, x, y, '-> ', end='')
  50. mx, my = moves[direction]
  51. x += mx
  52. y += my
  53. #print(direction, x, y, '-> ', end='')
  54. x = width - 2 if x == 0 else 1 if x == width - 1 else x
  55. y = height - 2 if y == 0 else 1 if y == height - 1 else y
  56. new_blizzards.add((direction, x, y))
  57. #print(direction, x, y)
  58. return new_blizzards
  59. def find_goal(blizzards, width, height, start, goal, visualize=False):
  60. time = 0
  61. positions = deque([start])
  62. while True:
  63. # visualize
  64. if visualize:
  65. # scale wait time to account for longer computation
  66. timepkg.sleep((500 - time) / 5000)
  67. print("\033c", end='')
  68. print(f'time is {time}')
  69. print_grid(blizzards, positions, width, height, start, goal)
  70. # move blizzards
  71. blizzards = simulate_blizzards(blizzards, width, height)
  72. new_positions = deque()
  73. for position in positions:
  74. # consider options and add new positions
  75. x, y = position
  76. options = [(x, y), (x - 1, y), (x, y - 1), (x + 1, y), (x, y + 1)]
  77. for option in options:
  78. ox, oy = option
  79. if option in new_positions:
  80. continue
  81. if option == goal:
  82. # move blizzards
  83. print(f'found goal in {time + 1} minutes')
  84. if visualize:
  85. timepkg.sleep(3)
  86. return blizzards, time + 1
  87. if (ox < 1 or oy < 1 or ox > width - 2 or oy > height - 2) and option != start:
  88. continue
  89. if ('>', ox, oy) in blizzards or ('v', ox, oy) in blizzards or ('<', ox, oy) in blizzards or ('^', ox, oy) in blizzards:
  90. continue
  91. new_positions.append(option)
  92. positions.clear()
  93. positions = new_positions
  94. time += 1
  95. if len(positions) == 0:
  96. print('error')
  97. return -1
  98. def tally_time(time, new_time):
  99. total_time = time + new_time
  100. print(f'new total time is {total_time} minutes')
  101. return total_time
  102. def solution(back_and_forth=False, visualize=False):
  103. total_time = 0
  104. blizzards, width, height = parse_input()
  105. start = (1, 0)
  106. goal = (width - 2, height - 1)
  107. # from start to finish
  108. blizzards, time = find_goal(blizzards, width, height, start, goal, visualize=visualize)
  109. total_time = tally_time(total_time, time)
  110. if not back_and_forth: return
  111. # from finish to start
  112. blizzards, time = find_goal(blizzards, width, height, goal, start, visualize=visualize)
  113. total_time = tally_time(total_time, time)
  114. # from start to finish
  115. blizzards, time = find_goal(blizzards, width, height, start, goal, visualize=visualize)
  116. total_time = tally_time(total_time, time)
  117. if sys.argv[1] in '1':
  118. solution(visualize=True)
  119. else:
  120. solution(back_and_forth=True, visualize=True)