From 6d9fbe0edf35d99642ebdfa95f582dc8c30c51c9 Mon Sep 17 00:00:00 2001 From: kageru Date: Fri, 10 Dec 2021 15:09:09 +0100 Subject: [PATCH] Refactor day 10 --- 2021/src/bin/day10.rs | 108 +++++++++++++++--------------------------- 1 file changed, 39 insertions(+), 69 deletions(-) diff --git a/2021/src/bin/day10.rs b/2021/src/bin/day10.rs index 87ae894..0192693 100644 --- a/2021/src/bin/day10.rs +++ b/2021/src/bin/day10.rs @@ -1,71 +1,46 @@ #![feature(test)] extern crate test; - use aoc2021::common::*; const DAY: usize = 10; -type Parsed<'a> = Vec<&'a str>; -fn parse_input(raw: &str) -> Parsed { - raw.lines().collect() -} - -fn solve(parsed: &Parsed) -> (usize, usize) { - let mut points = 0; - let mut p2_points = Vec::new(); - for &line in parsed { - let mut stack = Vec::new(); - let mut invalid = false; - for c in line.bytes() { - match c { - b'(' | b'[' | b'<' | b'{' => stack.push(c), - b')' if stack.last() == Some(&b'(') => { - stack.pop(); - } - b']' if stack.last() == Some(&b'[') => { - stack.pop(); - } - b'>' if stack.last() == Some(&b'<') => { - stack.pop(); - } - b'}' if stack.last() == Some(&b'{') => { - stack.pop(); - } - b')' => { - points += 3; - invalid = true; - break; - } - b']' => { - points += 57; - invalid = true; - break; - } - b'}' => { - points += 1197; - invalid = true; - break; - } - b'>' => { - points += 25137; - invalid = true; - break; - } - _ => unreachable!(), +fn solve(input: &str) -> (usize, usize) { + let mut p1_score = 0; + let mut p2_scores = Vec::new(); + let mut stack = Vec::new(); + for line in input.lines() { + match is_well_formed(line, &mut stack) { + Ok(s) => p2_scores.push(autocomplete_points(s)), // <- clears the stack internally + Err(p) => { + p1_score += p; + stack.clear(); } } - if invalid { - continue; - } - p2_points.push(autocomplete_points(stack)); } - let p2_len = p2_points.len(); - (points, *p2_points.select_nth_unstable(p2_len / 2).1) + let p2_len = p2_scores.len(); + (p1_score, *p2_scores.select_nth_unstable(p2_len / 2).1) } -fn autocomplete_points(stack: Vec) -> usize { +fn is_well_formed<'a>(line: &str, stack: &'a mut Vec) -> Result<&'a mut Vec, usize> { + for c in line.bytes() { + match (stack.last(), c) { + (_, b'(' | b'[' | b'<' | b'{') => stack.push(c), + (Some(b'('), b')') | (Some(b'['), b']') | (Some(b'{'), b'}') | (Some(b'<'), b'>') => { + stack.pop(); + } + (_, b')') => return Err(3), + (_, b']') => return Err(57), + (_, b'}') => return Err(1197), + (_, b'>') => return Err(25137), + _ => unreachable!(), + } + } + Ok(stack) +} + +fn autocomplete_points(stack: &mut Vec) -> usize { let mut points = 0; - for p in stack.into_iter().rev() { + for p in stack.drain(..).rev() { points *= 5; match p { b'(' => points += 1, @@ -80,8 +55,7 @@ fn autocomplete_points(stack: Vec) -> usize { fn main() { let raw = read_file(DAY); - let input = parse_input(&raw); - let (p1, p2) = solve(&input); + let (p1, p2) = solve(&raw); println!("Part 1: {p1}"); println!("Part 2: {p2}"); } @@ -102,29 +76,25 @@ mod tests { <{([([[(<>()){}]>(<<{{ <{([{{}}[<[[[<>{}]]]>[]]"; - #[test] - fn test_autocomplete_points() { - assert_eq!(autocomplete_points(vec![b'<', b'{', b'(', b'[']), 294); + #[bench] + fn bench_autocomplete_points(b: &mut test::Bencher) { + let sample_stack = vec![b'<', b'{', b'(', b'[']; + b.iter(|| assert_eq!(autocomplete_points(test::black_box(&mut sample_stack.clone())), 294)); } #[test] fn part1_test() { - let input = parse_input(TEST_INPUT); - assert_eq!(solve(&input).0, 26397); + assert_eq!(solve(TEST_INPUT).0, 26397); } #[test] fn part2_test() { - let input = parse_input(TEST_INPUT); - assert_eq!(solve(&input).1, 288957); + assert_eq!(solve(TEST_INPUT).1, 288957); } #[bench] fn bench_solution(b: &mut test::Bencher) { let raw = read_file(DAY); - let input = parse_input(&raw); - b.iter(|| assert_eq!(solve(test::black_box(&input)), (358737, 4329504793))) + b.iter(|| assert_eq!(solve(test::black_box(&raw)), (358737, 4329504793))) } - - bench_input!(Vec::len => 90); }