do day 21

This commit is contained in:
Adam Jeniski 2022-12-29 13:55:11 -05:00
parent fe0345c16f
commit e3f494b093
4 changed files with 1719 additions and 0 deletions

1551
2022/input/day_21.txt Normal file

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,15 @@
root: pppw + sjmn
dbpl: 5
cczh: sllz + lgvd
zczc: 2
ptdq: humn - dvpt
dvpt: 3
lfqf: 4
humn: 5
ljgn: 2
sjmn: drzm * dbpl
sllz: 4
pppw: cczh / lfqf
lgvd: ljgn * ptdq
drzm: hmdt - zczc
hmdt: 32

152
2022/src/day_21.rs Normal file
View File

@ -0,0 +1,152 @@
use std::collections::HashMap;
use std::str::FromStr;
use Monkey::*;
use Operation::*;
#[derive(Clone, Copy, Debug)]
enum Operation {
Add,
Subtract,
Multiply,
Divide,
}
#[derive(Clone, Debug)]
enum Monkey {
Number(String, i64),
Pending(String, String, Operation, String),
}
impl FromStr for Monkey {
type Err = ();
fn from_str(s: &str) -> Result<Self, Self::Err> {
let (name, rest) = s.split_once(": ").unwrap();
if let Ok(num) = rest.parse::<i64>() {
return Ok(Number(name.into(), num));
}
let (name_left, rest) = rest.split_once(' ').unwrap();
let (operation, name_right) = rest.split_once(' ').unwrap();
return Ok(Pending(
name.into(),
name_left.into(),
match operation {
"+" => Add,
"-" => Subtract,
"*" => Multiply,
"/" => Divide,
_ => panic!("expected math symbol"),
},
name_right.into(),
));
}
}
type Data = HashMap<String, Monkey>;
type Output = i64;
fn parse(input: &str) -> Data {
input
.lines()
.map(|s| s.parse::<Monkey>().unwrap())
.map(|m| {
(
match m.clone() {
Number(name, _) => name,
Pending(name, _, _, _) => name,
}
.to_string(),
m,
)
})
.collect()
}
fn part_one(mut data: Data) -> Output {
fn eval(map: &mut Data, name: &str) -> i64 {
let node = map.get(name).unwrap().clone();
match node {
Number(_, num) => num,
Pending(name, left, op, right) => {
let left = eval(map, left.as_ref());
let right = eval(map, right.as_ref());
let res = match op {
Add => left + right,
Subtract => left - right,
Multiply => left * right,
Divide => left / right,
};
*map.get_mut(&name).unwrap() = Number(name.clone(), res);
res
}
}
}
eval(&mut data, "root")
}
fn part_two(mut data: Data) -> Output {
fn eval(map: &mut Data, name: &str) -> Option<i64> {
if name == "humn" {
return None;
}
let node = map.get(name).unwrap().clone();
match node {
Number(_, num) => Some(num),
Pending(name, left, op, right) => {
let left = eval(map, left.as_ref())?;
let right = eval(map, right.as_ref())?;
let res = match op {
Add => left + right,
Subtract => left - right,
Multiply => left * right,
Divide => left / right,
};
*map.get_mut(&name).unwrap() = Number(name.clone(), res);
Some(res)
}
}
}
fn find_humn(map: &mut Data, name: &str, equals: i64) -> i64 {
if name == "humn" {
return equals;
}
let curr = map.get(name).unwrap().clone();
match curr {
Pending(_, left, op, right) => {
if let Some(n) = eval(map, left.as_str()) {
match op {
Add => find_humn(map, right.as_str(), equals - n),
Subtract => find_humn(map, right.as_str(), n - equals),
Multiply => find_humn(map, right.as_str(), equals / n),
Divide => find_humn(map, right.as_str(), n / equals),
}
} else {
let n = eval(map, right.as_ref()).unwrap();
match op {
Add => find_humn(map, left.as_str(), equals - n),
Subtract => find_humn(map, left.as_str(), equals + n),
Multiply => find_humn(map, left.as_str(), equals / n),
Divide => find_humn(map, left.as_str(), equals * n),
}
}
}
_ => panic!("expected to do pending"),
}
}
let root = data.get("root").unwrap().clone();
match root {
Pending(_, left, _, right) => {
if let Some(n) = eval(&mut data, left.as_str()) {
find_humn(&mut data, right.as_str(), n)
} else {
let n = eval(&mut data, right.as_str()).unwrap();
find_humn(&mut data, left.as_str(), n)
}
}
_ => panic!(),
}
}
advent_of_code_macro::generate_tests!(
day 21,
parse,
part_one,
part_two,
sample tests [152, 301],
star tests [232_974_643_455_000, 3_740_214_169_961]
);

View File

@ -19,3 +19,4 @@ mod day_17;
mod day_18; mod day_18;
mod day_19; mod day_19;
mod day_20; mod day_20;
mod day_21;