aoc2024/src/solutions/day_3.rs
2024-12-06 00:55:56 +07:00

55 lines
1.5 KiB
Rust

use std::{fs::read_to_string, path::Path, sync::LazyLock};
use regex::Regex;
use super::AocSolution;
use crate::utils::Result;
static MUL: LazyLock<Regex> =
LazyLock::new(|| Regex::new(r"mul\([0-9]{1,3},[0-9]{1,3}\)").unwrap());
static COND_MUL: LazyLock<Regex> =
LazyLock::new(|| Regex::new(r"(mul\([0-9]{1,3},[0-9]{1,3}\))|(do\(\))|(don't\(\))").unwrap());
fn as_mul(s: &str) -> u64 {
let (a, b) = s[4..(s.len() - 1)]
.split_once(',')
.map(|(a, b)| (a.parse::<u64>().unwrap(), b.parse::<u64>().unwrap()))
.unwrap();
a * b
}
#[derive(Clone, Copy)]
pub struct AocDayThreeSolution;
impl AocSolution for AocDayThreeSolution {
type Output = u64;
fn get_input(&self, path: Option<&Path>) -> Result<String> {
Ok(match path {
Some(p) => read_to_string(p)?,
None => read_to_string("./input/day_3.txt")?,
})
}
fn part_one(&self, input: &str) -> Result<Self::Output> {
Ok(MUL.find_iter(input).map(|x| x.as_str()).map(as_mul).sum())
}
fn part_two(&self, input: &str) -> Result<Self::Output> {
Ok(COND_MUL
.find_iter(input)
.map(|x| x.as_str())
.fold((0, true), |(sum, flag), cut| {
if cut.starts_with("mul") && flag {
(sum + as_mul(cut), flag)
} else if cut == "do()" {
(sum, true)
} else {
(sum, false)
}
})
.0)
}
}