Skip to content

Commit aa36f15

Browse files
committed
[ADD] day21: twenty-first day solutions
1 parent 3b964e7 commit aa36f15

File tree

4 files changed

+120
-12
lines changed

4 files changed

+120
-12
lines changed

README.md

+12-12
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,21 @@
11
# Advent of Code 2021 🎅🏻
22

33
[![About](https://img.shields.io/badge/Advent%20of%20Code%20🎄-2021-brightgreen)](https://adventofcode.com/2021/about)
4-
[![Days completed](https://img.shields.io/badge/day%20📅-20-blue)](https://adventofcode.com/2021)
5-
[![Stars](https://img.shields.io/badge/stars%20⭐-40-yellow)](https://adventofcode.com/2021/stats)
4+
[![Days completed](https://img.shields.io/badge/day%20📅-21-blue)](https://adventofcode.com/2021)
5+
[![Stars](https://img.shields.io/badge/stars%20⭐-42-yellow)](https://adventofcode.com/2021/stats)
66

77
#### I'll be able to solve the puzzles !?
88

9-
| Day | Language | Solutions | Day | Language | Solutions |
10-
|-----------------|------------|-----------------------|-----------------|------------|---------------------|
11-
| [day01](day01/) | **Python** | 1564, 1611 | [day14](day14/) | **Python** | 2010, 2437698971143 |
12-
| [day02](day02/) | **Python** | 1714950, 1281977850 | [day15](day15/) | **Python** | 595, 2914 |
13-
| [day03](day03/) | **Python** | 1997414, 1032597 | [day16](day16/) | **Python** | 877, 194435634456 |
14-
| [day04](day04/) | **Python** | 67716, 1830 | [day17](day17/) | **Python** | 3160, 1928 |
15-
| [day05](day05/) | **Python** | 5608, 20299 | [day18](day18/) | **Python** | 3665, 4775 |
16-
| [day06](day06/) | **Python** | 372300, 1675781200288 | [day19](day19/) | **Python** | 396, 11828 |
17-
| [day07](day07/) | **Python** | 347509, 98257206 | [day20](day20/) | **Python** | 5479, 19012 |
18-
| [day08](day08/) | **Python** | 303, 961734 |
9+
| Day | Language | Solutions | Day | Language | Solutions |
10+
|-----------------|------------|-----------------------|-----------------|------------|-------------------------|
11+
| [day01](day01/) | **Python** | 1564, 1611 | [day14](day14/) | **Python** | 2010, 2437698971143 |
12+
| [day02](day02/) | **Python** | 1714950, 1281977850 | [day15](day15/) | **Python** | 595, 2914 |
13+
| [day03](day03/) | **Python** | 1997414, 1032597 | [day16](day16/) | **Python** | 877, 194435634456 |
14+
| [day04](day04/) | **Python** | 67716, 1830 | [day17](day17/) | **Python** | 3160, 1928 |
15+
| [day05](day05/) | **Python** | 5608, 20299 | [day18](day18/) | **Python** | 3665, 4775 |
16+
| [day06](day06/) | **Python** | 372300, 1675781200288 | [day19](day19/) | **Python** | 396, 11828 |
17+
| [day07](day07/) | **Python** | 347509, 98257206 | [day20](day20/) | **Python** | 5479, 19012 |
18+
| [day08](day08/) | **Python** | 303, 961734 | [day21](day21/) | **Python** | 752745, 309196008717909 |
1919
| [day09](day09/) | **Python** | 535, 1122700 |
2020
| [day10](day10/) | **Python** | 167379, 2776842859 |
2121
| [day11](day11/) | **Python** | 1585, 382 |

day21/input.txt

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
Player 1 starting position: 6
2+
Player 2 starting position: 3

day21/solution.py

+84
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
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)

day21/test_solution.py

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
# -*- coding: utf-8 -*-
2+
3+
import unittest
4+
5+
from solution import get_players_from_input, get_total_points_losing_player, get_winner_in_more_universes
6+
7+
input_test = """Player 1 starting position: 4
8+
Player 2 starting position: 8"""
9+
10+
11+
class TestSolution(unittest.TestCase):
12+
13+
def setUp(self):
14+
self.player1, self.player2 = get_players_from_input(input_test)
15+
16+
def test_get_total_points_losing_player(self):
17+
count = get_total_points_losing_player(self.player1, self.player2)
18+
self.assertEqual(count, 739785)
19+
20+
def test_get_winner_in_more_universes(self):
21+
count = get_winner_in_more_universes(self.player1, self.player2)
22+
self.assertEqual(count, 444356092776315)

0 commit comments

Comments
 (0)