advent-of-code/2019/14/src/main.rs
2019-12-15 09:20:33 +01:00

106 lines
3.1 KiB
Rust

#[macro_use]
extern crate lazy_static;
use std::collections::HashMap;
use std::io::{stdin, BufRead};
use std::sync::Mutex;
#[derive(Debug)]
struct Element<'a> {
quantity: usize,
element: &'a str,
}
#[derive(Debug)]
struct Reaction<'a> {
inputs: Vec<Element<'a>>,
output: Element<'a>,
}
impl<'a> From<&'a str> for Element<'a> {
fn from(other: &'a str) -> Self {
let mut parts = other.split(' ');
Element {
quantity: parts.next().unwrap().parse().unwrap(),
element: parts.next().unwrap(),
}
}
}
impl<'a> From<&'a str> for Reaction<'a> {
fn from(other: &'a str) -> Self {
let in_out: Vec<_> = other.split(" => ").collect();
let inputs = in_out[0].split(", ").map(|p| p.into()).collect();
let output = in_out[1].into();
Reaction { inputs, output }
}
}
lazy_static! {
static ref LINES: Vec<String> = stdin().lock().lines().map(|l| l.unwrap()).collect();
static ref REACTIONS: HashMap<&'static str, Reaction<'static>> = LINES
.iter()
.map::<Reaction, _>(|l| l.as_str().into())
.map(|r| (r.output.element.clone(), r))
.collect();
static ref STORAGE: Mutex<HashMap<&'static str, usize>> = Mutex::new(HashMap::new());
}
const ORE: &str = "ORE";
const FUEL: &str = "FUEL";
fn main() {
let p1 = count_ingredients(FUEL, 1);
println!("Part 1: {}", p1);
println!("Part 2: {}", make_all_the_fuel(p1, 1_000_000_000_000));
}
// Not beautiful, but it gets the job done.
fn make_all_the_fuel(mut total_ore: usize, limit: usize) -> usize {
let initial_batch_size = 1 << 15;
let mut batch_size = initial_batch_size;
let mut fuel = 0;
while total_ore < limit {
total_ore += count_ingredients(FUEL, batch_size);
fuel += batch_size;
// gib int multiplication with float pls
if total_ore > limit - (limit / (initial_batch_size / batch_size)) && batch_size > 1 {
batch_size /= 2;
}
}
fuel
}
fn count_ingredients(element: &'static str, desired_quantity: usize) -> usize {
if element == ORE {
return desired_quantity;
}
let mut ores = 0;
let reaction = REACTIONS.get(&element).unwrap();
let output_quantity = reaction.output.quantity;
let num_reactions = (desired_quantity as f32 / output_quantity as f32).ceil() as usize;
for input in &reaction.inputs {
let needed = input.quantity * num_reactions;
let in_storage = get_from_storage(input.element);
if in_storage >= needed {
put_into_storage(input.element, in_storage - needed);
} else {
put_into_storage(input.element, 0);
ores += count_ingredients(input.element, needed - in_storage);
}
}
let surplus = (output_quantity * num_reactions) - desired_quantity;
put_into_storage(element, surplus);
ores
}
#[inline]
fn get_from_storage(element: &str) -> usize {
*STORAGE.lock().unwrap().get(element).unwrap_or(&0)
}
#[inline]
fn put_into_storage(element: &'static str, quantity: usize) {
STORAGE.lock().unwrap().insert(element, quantity);
}