|
| 1 | +# -*- coding: utf-8 -*- |
| 2 | + |
| 3 | +from itertools import cycle |
| 4 | + |
| 5 | + |
| 6 | +def get_players_from_input(data): |
| 7 | + player1, player2 = [int(line[-2:]) for line in data.split("\n")] |
| 8 | + return player1, player2 |
| 9 | + |
| 10 | + |
| 11 | +def read_file(filename): |
| 12 | + with open(filename) as file: |
| 13 | + data = file.read() |
| 14 | + return get_players_from_input(data) |
| 15 | + |
| 16 | + |
| 17 | +def get_deterministic_die(): |
| 18 | + return cycle(range(1, 101)) |
| 19 | + |
| 20 | + |
| 21 | +def perform_game(player1, player2, die, score1, score2, throws): |
| 22 | + if score2 >= 1000: |
| 23 | + return score1 * throws |
| 24 | + player1 += sum(next(die) for _ in range(3)) |
| 25 | + player1 = (player1 - 1) % 10 + 1 |
| 26 | + score1 += player1 |
| 27 | + return perform_game(player2, player1, die, score2, score1, throws + 3) |
| 28 | + |
| 29 | + |
| 30 | +def get_total_points_losing_player(player1, player2, score1=0, score2=0, throws=0): |
| 31 | + deterministic_die = cycle(range(1, 101)) |
| 32 | + total_points = perform_game(player1, player2, deterministic_die, score1, score2, throws) |
| 33 | + return total_points |
| 34 | + |
| 35 | + |
| 36 | +def get_roll_sums(): |
| 37 | + three_roll_sums = [] |
| 38 | + for one in (1, 2, 3): |
| 39 | + for two in (1, 2, 3): |
| 40 | + for three in (1, 2, 3): |
| 41 | + three_roll_sums.append(one + two + three) |
| 42 | + return three_roll_sums |
| 43 | + |
| 44 | + |
| 45 | +def get_winner_in_more_universes(player1, player2, score1=0, score2=0): |
| 46 | + three_roll_sums = get_roll_sums() |
| 47 | + player_one_wins = 0 |
| 48 | + player_two_wins = 0 |
| 49 | + game_initial = (player1 - 1, score1, player2 - 1, score2, 1) |
| 50 | + games = {game_initial: 1} |
| 51 | + while games: |
| 52 | + new_games = {} |
| 53 | + for game, universes in games.items(): |
| 54 | + throws = game[4] |
| 55 | + if throws == 1: |
| 56 | + pos, score = game[0], game[1] |
| 57 | + else: |
| 58 | + pos, score = game[2], game[3] |
| 59 | + for roll_sum in three_roll_sums: |
| 60 | + new_position = (pos + roll_sum) % 10 |
| 61 | + new_score = score + new_position + 1 |
| 62 | + if new_score >= 21: |
| 63 | + if throws == 1: |
| 64 | + player_one_wins += universes |
| 65 | + else: |
| 66 | + player_two_wins += universes |
| 67 | + else: |
| 68 | + if throws == 1: |
| 69 | + new_value = (new_position, new_score, game[2], game[3], 2) |
| 70 | + else: |
| 71 | + new_value = (game[0], game[1], new_position, new_score, 1) |
| 72 | + new_games[new_value] = new_games.get(new_value, 0) + universes |
| 73 | + games = new_games |
| 74 | + return max(player_one_wins, player_two_wins) |
| 75 | + |
| 76 | + |
| 77 | +if __name__ == "__main__": |
| 78 | + player1, player2 = read_file("input.txt") |
| 79 | + |
| 80 | + first_solution = get_total_points_losing_player(player1, player2) |
| 81 | + print(first_solution) |
| 82 | + |
| 83 | + second_solution = get_winner_in_more_universes(player1, player2) |
| 84 | + print(second_solution) |
0 commit comments