mirror of
https://github.com/Ajetski/advent-of-code.git
synced 2025-09-30 07:23:18 -09:00
do day 21
This commit is contained in:
parent
fe0345c16f
commit
e3f494b093
1551
2022/input/day_21.txt
Normal file
1551
2022/input/day_21.txt
Normal file
File diff suppressed because it is too large
Load Diff
15
2022/input/day_21_sample.txt
Normal file
15
2022/input/day_21_sample.txt
Normal 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
152
2022/src/day_21.rs
Normal 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]
|
||||
);
|
@ -19,3 +19,4 @@ mod day_17;
|
||||
mod day_18;
|
||||
mod day_19;
|
||||
mod day_20;
|
||||
mod day_21;
|
||||
|
Loading…
x
Reference in New Issue
Block a user