day 11
parent
db450943a0
commit
2b36afe92b
@ -0,0 +1,42 @@
|
||||
# This file is automatically @generated by Cargo.
|
||||
# It is not intended for manual editing.
|
||||
version = 3
|
||||
|
||||
[[package]]
|
||||
name = "aho-corasick"
|
||||
version = "0.7.20"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "cc936419f96fa211c1b9166887b38e5e40b19958e5b895be7c1f93adec7071ac"
|
||||
dependencies = [
|
||||
"memchr",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "day11"
|
||||
version = "0.1.0"
|
||||
dependencies = [
|
||||
"regex",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "memchr"
|
||||
version = "2.5.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "2dffe52ecf27772e601905b7522cb4ef790d2cc203488bbd0e2fe85fcb74566d"
|
||||
|
||||
[[package]]
|
||||
name = "regex"
|
||||
version = "1.7.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e076559ef8e241f2ae3479e36f97bd5741c0330689e217ad51ce2c76808b868a"
|
||||
dependencies = [
|
||||
"aho-corasick",
|
||||
"memchr",
|
||||
"regex-syntax",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "regex-syntax"
|
||||
version = "0.6.28"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "456c603be3e8d448b072f410900c09faf164fbce2d480456f50eea6e25f9c848"
|
@ -0,0 +1,9 @@
|
||||
[package]
|
||||
name = "day11"
|
||||
version = "0.1.0"
|
||||
edition = "2021"
|
||||
|
||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||
|
||||
[dependencies]
|
||||
regex = "1"
|
@ -0,0 +1,189 @@
|
||||
use regex::Regex;
|
||||
use std::collections::{HashMap, VecDeque};
|
||||
|
||||
struct Monke {
|
||||
operation: Box<dyn Fn(usize) -> usize>,
|
||||
test: usize,
|
||||
result_recipient: (usize, usize),
|
||||
items: VecDeque<usize>,
|
||||
}
|
||||
|
||||
enum Star {
|
||||
Silver,
|
||||
Gold,
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let mut monkes = HashMap::new();
|
||||
|
||||
let s = std::fs::read_to_string("/dev/stdin").unwrap();
|
||||
let thing = s.split("\n\n");
|
||||
|
||||
let mut monke_magic_number = 1;
|
||||
for monke_str in thing {
|
||||
let (monke, id) = monke_parser(monke_str);
|
||||
monke_magic_number *= monke.test;
|
||||
monkes.insert(id, monke);
|
||||
}
|
||||
|
||||
let mut inspections_silver: HashMap<usize, usize> = HashMap::new();
|
||||
let mut inspections_gold = inspections_silver.clone();
|
||||
|
||||
monke_processor(&mut monkes, &mut inspections_silver, Star::Silver, 0);
|
||||
monke_processor(
|
||||
&mut monkes,
|
||||
&mut inspections_gold,
|
||||
Star::Gold,
|
||||
monke_magic_number,
|
||||
);
|
||||
|
||||
println!(
|
||||
"P1: {}, P2: {}",
|
||||
calc_monke_business(&inspections_silver),
|
||||
calc_monke_business(&inspections_gold)
|
||||
);
|
||||
}
|
||||
|
||||
fn calc_monke_business(inspections: &HashMap<usize, usize>) -> usize {
|
||||
let mut ins = inspections.iter().map(|v| v.1).collect::<Vec<_>>();
|
||||
ins.sort();
|
||||
ins.pop().unwrap() * ins.pop().unwrap()
|
||||
}
|
||||
|
||||
fn monke_processor(
|
||||
monkes: &mut HashMap<usize, Monke>,
|
||||
inspections: &mut HashMap<usize, usize>,
|
||||
part: Star,
|
||||
monke_magic_number: usize,
|
||||
) {
|
||||
let num_turns = if matches!(part, Star::Gold) {
|
||||
10_000
|
||||
} else {
|
||||
20
|
||||
};
|
||||
|
||||
for _ in 0..num_turns {
|
||||
for id in 0..monkes.len() {
|
||||
let monke = monkes.get_mut(&id).unwrap();
|
||||
let mut pass_queue = Vec::new();
|
||||
|
||||
let mut inspections_for_monke = *inspections.get(&id).unwrap_or(&0);
|
||||
|
||||
while let Some(mut worry_level) = monke.items.pop_front() {
|
||||
inspections_for_monke += 1;
|
||||
worry_level = (monke.operation)(worry_level);
|
||||
|
||||
match part {
|
||||
Star::Silver => worry_level /= 3,
|
||||
Star::Gold => worry_level %= monke_magic_number,
|
||||
}
|
||||
|
||||
let passes_test = worry_level % monke.test == 0;
|
||||
let recipient_id = if passes_test {
|
||||
monke.result_recipient.0
|
||||
} else {
|
||||
monke.result_recipient.1
|
||||
};
|
||||
pass_queue.push((recipient_id, worry_level));
|
||||
}
|
||||
|
||||
inspections.insert(id, inspections_for_monke);
|
||||
|
||||
for (recipient_id, val) in pass_queue {
|
||||
let recipient_monke = monkes.get_mut(&recipient_id).unwrap();
|
||||
recipient_monke.items.push_back(val);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn monke_parser(monke_str: &str) -> (Monke, usize) {
|
||||
let mut id = 0;
|
||||
let mut items = VecDeque::default();
|
||||
let mut operation: Box<dyn Fn(usize) -> usize>;
|
||||
let mut test = 0;
|
||||
let mut result_recipient = (0, 0);
|
||||
operation = Box::new(|n| n);
|
||||
|
||||
for mut line in monke_str.lines() {
|
||||
line = line.trim();
|
||||
if line.starts_with("Monkey") {
|
||||
id = Regex::new(r"\d+")
|
||||
.unwrap()
|
||||
.find(line)
|
||||
.unwrap()
|
||||
.as_str()
|
||||
.parse::<usize>()
|
||||
.unwrap();
|
||||
}
|
||||
if line.starts_with("Starting") {
|
||||
items = Regex::new(r"\d+(\d+, )?")
|
||||
.unwrap()
|
||||
.find_iter(line)
|
||||
.map(|item| item.as_str().parse::<usize>().unwrap())
|
||||
.collect::<VecDeque<usize>>();
|
||||
}
|
||||
if line.starts_with("Operation") {
|
||||
let op = Regex::new(r"[*+/-].+")
|
||||
.unwrap()
|
||||
.find(line)
|
||||
.unwrap()
|
||||
.as_str()
|
||||
.split_once(' ')
|
||||
.unwrap();
|
||||
|
||||
operation = match op.1 {
|
||||
"old" => match op.0 {
|
||||
"+" => Box::new(|n| n + n),
|
||||
"*" => Box::new(|n| n * n),
|
||||
_ => panic!("https://music.portal2sounds.com/2627"),
|
||||
},
|
||||
_ => {
|
||||
let num = op.1.parse::<usize>().unwrap();
|
||||
match op.0 {
|
||||
"+" => Box::new(move |n| n + num),
|
||||
"*" => Box::new(move |n| n * num),
|
||||
_ => panic!("https://music.portal2sounds.com/2627"),
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
if line.starts_with("Test") {
|
||||
test = Regex::new(r"[0-9]+")
|
||||
.unwrap()
|
||||
.find(line)
|
||||
.unwrap()
|
||||
.as_str()
|
||||
.parse::<usize>()
|
||||
.unwrap();
|
||||
}
|
||||
if line.starts_with("If true") {
|
||||
result_recipient.0 = Regex::new(r"[0-9]")
|
||||
.unwrap()
|
||||
.find(line)
|
||||
.unwrap()
|
||||
.as_str()
|
||||
.parse::<usize>()
|
||||
.unwrap();
|
||||
}
|
||||
if line.starts_with("If false") {
|
||||
result_recipient.1 = Regex::new(r"[0-9]")
|
||||
.unwrap()
|
||||
.find(line)
|
||||
.unwrap()
|
||||
.as_str()
|
||||
.parse::<usize>()
|
||||
.unwrap();
|
||||
}
|
||||
}
|
||||
|
||||
(
|
||||
Monke {
|
||||
items,
|
||||
operation,
|
||||
test,
|
||||
result_recipient,
|
||||
},
|
||||
id,
|
||||
)
|
||||
}
|
Loading…
Reference in New Issue