mirror of
https://github.com/bertptrs/adventofcode.git
synced 2025-12-25 21:00:31 +01:00
172 lines
4.0 KiB
Rust
172 lines
4.0 KiB
Rust
use std::io::Read;
|
|
|
|
use crate::common::LineIter;
|
|
|
|
fn read_input(input: &mut dyn Read) -> (i32, i32) {
|
|
let mut reader = LineIter::new(input);
|
|
|
|
let mut helper = || {
|
|
reader
|
|
.next()
|
|
.unwrap()
|
|
.split(' ')
|
|
.last()
|
|
.unwrap()
|
|
.parse()
|
|
.unwrap()
|
|
};
|
|
|
|
let a = helper();
|
|
let b = helper();
|
|
|
|
(a, b)
|
|
}
|
|
|
|
#[inline]
|
|
fn simulate(die: i32, pos: i32) -> i32 {
|
|
(pos + 3 * die + 3 - 1) % 10 + 1
|
|
}
|
|
|
|
fn find_repetition(mut pos: i32, mut die: i32) -> i32 {
|
|
let mut advance = 0;
|
|
|
|
for _ in 0..10 {
|
|
pos = simulate(die, pos);
|
|
advance += pos;
|
|
die = (die + 6) % 10;
|
|
}
|
|
|
|
advance
|
|
}
|
|
|
|
pub fn part1(input: &mut dyn Read) -> String {
|
|
const TARGET_SCORE: i32 = 1000;
|
|
|
|
let (mut a, mut b) = read_input(input);
|
|
|
|
let a10 = find_repetition(a, 1);
|
|
let b10 = find_repetition(b, 4);
|
|
|
|
let a_win = TARGET_SCORE / a10;
|
|
let b_win = TARGET_SCORE / b10;
|
|
|
|
let win = a_win.min(b_win);
|
|
let mut a_score = win * a10;
|
|
let mut b_score = win * b10;
|
|
let mut die = 1;
|
|
let mut rolls = 3 * 20 * win;
|
|
|
|
let (loser_score, rolls) = if a_win < b_win {
|
|
while a_score < TARGET_SCORE {
|
|
a = simulate(die, a);
|
|
a_score += a;
|
|
rolls += 3;
|
|
|
|
if a_score < TARGET_SCORE {
|
|
b = simulate(die + 3, b);
|
|
b_score += b;
|
|
rolls += 3;
|
|
}
|
|
|
|
die = (die + 6) % 10;
|
|
}
|
|
|
|
(b_score, rolls)
|
|
} else {
|
|
while b_score < TARGET_SCORE {
|
|
a = simulate(die, a);
|
|
a_score += a;
|
|
|
|
b = simulate(die + 3, b);
|
|
b_score += b;
|
|
|
|
rolls += 6;
|
|
|
|
die = (die + 6) % 10;
|
|
}
|
|
|
|
(a_score, rolls)
|
|
};
|
|
|
|
(loser_score * rolls).to_string()
|
|
}
|
|
|
|
const MAX_TURNS: usize = 13;
|
|
const ROLLS: [i32; 7] = [3, 4, 5, 6, 7, 8, 9];
|
|
const FREQS: [u64; 7] = [1, 3, 6, 7, 6, 3, 1];
|
|
|
|
fn multiverse(pos: i32) -> ([u64; MAX_TURNS], [u64; MAX_TURNS]) {
|
|
type State = [[u64; 10]; 21];
|
|
|
|
// Let's work with pos - 1 as pos for now, indexes nicer;
|
|
let pos = pos as usize - 1;
|
|
|
|
let mut alive = [0; MAX_TURNS];
|
|
let mut wins = [0; MAX_TURNS];
|
|
|
|
alive[0] = 1;
|
|
|
|
let mut current = [[0u64; 10]; 21];
|
|
current[0][pos] = 1;
|
|
let mut next = [[0u64; 10]; 21];
|
|
|
|
let mut helper = |turn, current: &State, next: &mut State| {
|
|
next.iter_mut().flatten().for_each(|v| *v = 0);
|
|
|
|
for (score, score_row) in current.iter().enumerate() {
|
|
for (pos, &pop) in score_row.iter().enumerate() {
|
|
for (roll, freq) in ROLLS.into_iter().zip(FREQS) {
|
|
let new_pos = (pos + (roll as usize)) % 10;
|
|
let new_score = score + new_pos + 1; // +1 because above
|
|
let new_pop = freq * pop;
|
|
|
|
if new_score >= next.len() {
|
|
wins[turn] += new_pop;
|
|
} else {
|
|
alive[turn] += new_pop;
|
|
next[new_score][new_pos] += new_pop;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
for turn in (1..MAX_TURNS).step_by(2) {
|
|
helper(turn, ¤t, &mut next);
|
|
helper(turn + 1, &next, &mut current);
|
|
}
|
|
|
|
(wins, alive)
|
|
}
|
|
|
|
pub fn part2(input: &mut dyn Read) -> String {
|
|
let (a, b) = read_input(input);
|
|
|
|
let (a_wins, a_alive) = multiverse(a);
|
|
let (b_wins, b_alive) = multiverse(b);
|
|
|
|
let a_winner: u64 = a_wins[1..].iter().zip(b_alive).map(|(&a, b)| a * b).sum();
|
|
let b_winner: u64 = b_wins.into_iter().zip(a_alive).map(|(a, b)| a * b).sum();
|
|
|
|
a_winner.max(b_winner).to_string()
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use crate::test_implementation;
|
|
|
|
use super::*;
|
|
|
|
const SAMPLE: &[u8] = include_bytes!("samples/21.txt");
|
|
|
|
#[test]
|
|
fn sample_part1() {
|
|
test_implementation(part1, SAMPLE, 739785);
|
|
}
|
|
|
|
#[test]
|
|
fn sample_part2() {
|
|
test_implementation(part2, SAMPLE, 444356092776315u64);
|
|
}
|
|
}
|