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); } }