Day 01
This commit is contained in:
parent
8c58627184
commit
5f04334b3d
5 changed files with 2187 additions and 0 deletions
16
Cargo.toml
Normal file
16
Cargo.toml
Normal file
|
@ -0,0 +1,16 @@
|
|||
[package]
|
||||
name = "executor"
|
||||
version = "0.0.0"
|
||||
authors = ["Anthony Cicchetti <anthony@anthonycicchetti.com>"]
|
||||
edition = "2021"
|
||||
|
||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||
|
||||
[dependencies]
|
||||
clap = "2.33.3"
|
||||
day_01 = { path = "./day_01" }
|
||||
|
||||
[workspace]
|
||||
members = [
|
||||
'day_01',
|
||||
]
|
2000
data/day_1_input.txt
Normal file
2000
data/day_1_input.txt
Normal file
File diff suppressed because it is too large
Load diff
9
day_01/Cargo.toml
Normal file
9
day_01/Cargo.toml
Normal file
|
@ -0,0 +1,9 @@
|
|||
[package]
|
||||
name = "day_01"
|
||||
version = "0.1.0"
|
||||
edition = "2021"
|
||||
|
||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||
|
||||
[dependencies]
|
||||
itertools = "*"
|
110
day_01/src/lib.rs
Normal file
110
day_01/src/lib.rs
Normal file
|
@ -0,0 +1,110 @@
|
|||
use itertools::Itertools;
|
||||
|
||||
fn input_parse(input: &str) -> Vec<u32> {
|
||||
input
|
||||
.split_whitespace()
|
||||
.map(|t|
|
||||
t.parse::<u32>()
|
||||
.expect(&format!("Could not parse {} into u32", &t))
|
||||
)
|
||||
.collect::<Vec<_>>()
|
||||
}
|
||||
|
||||
/// As the submarine drops below the surface of the ocean, it automatically performs a sonar sweep of the nearby sea floor. On a small screen, the sonar sweep report (your puzzle input) appears: each line is a measurement of the sea floor depth as the sweep looks further and further away from the submarine.
|
||||
///
|
||||
/// For example, suppose you had the following report:
|
||||
///
|
||||
/// 199
|
||||
/// 200
|
||||
/// 208
|
||||
/// 210
|
||||
/// 200
|
||||
/// 207
|
||||
/// 240
|
||||
/// 269
|
||||
/// 260
|
||||
/// 263
|
||||
///
|
||||
/// This report indicates that, scanning outward from the submarine, the sonar sweep found depths of 199, 200, 208, 210, and so on.
|
||||
///
|
||||
/// The first order of business is to figure out how quickly the depth increases, just so you know what you're dealing with - you never know if the keys will get carried into deeper water by an ocean current or a fish or something.
|
||||
///
|
||||
/// To do this, count the number of times a depth measurement increases from the previous measurement. (There is no measurement before the first measurement.) In the example above, the changes are as follows:
|
||||
///
|
||||
/// > 199 (N/A - no previous measurement)
|
||||
/// >
|
||||
/// > 200 (**increased**)
|
||||
/// >
|
||||
/// > 208 (**increased**)
|
||||
/// >
|
||||
/// > 210 (**increased**)
|
||||
/// >
|
||||
/// > 200 (decreased)
|
||||
/// >
|
||||
/// > 207 (**increased**)
|
||||
/// >
|
||||
/// > 240 (**increased**)
|
||||
/// >
|
||||
/// > 269 (**increased**)
|
||||
/// >
|
||||
/// > 260 (decreased)
|
||||
/// >
|
||||
/// > 263 (**increased**)
|
||||
///
|
||||
/// In this example, there are 7 measurements that are larger than the previous measurement.
|
||||
///
|
||||
pub fn part1(input: &str) -> u32 {
|
||||
let input = input_parse(input);
|
||||
|
||||
input.iter()
|
||||
.tuple_windows::<(_,_)>()
|
||||
.fold(0, |acc, inp| { if inp.0 < inp.1 { acc + 1 } else { acc }})
|
||||
}
|
||||
|
||||
///
|
||||
/// Considering every single measurement isn't as useful as you expected: there's just too much noise in the data.
|
||||
///
|
||||
/// Instead, consider sums of a three-measurement sliding window. Again considering the above example:
|
||||
///
|
||||
/// ```text
|
||||
/// 199 A
|
||||
/// 200 A B
|
||||
/// 208 A B C
|
||||
/// 210 B C D
|
||||
/// 200 E C D
|
||||
/// 207 E F D
|
||||
/// 240 E F G
|
||||
/// 269 F G H
|
||||
/// 260 G H
|
||||
/// 263 H
|
||||
/// ```
|
||||
///
|
||||
/// Start by comparing the first and second three-measurement windows. The measurements in the first window are marked A (199, 200, 208); their sum is 199 + 200 + 208 = 607. The second window is marked B (200, 208, 210); its sum is 618. The sum of measurements in the second window is larger than the sum of the first, so this first comparison increased.
|
||||
///
|
||||
/// Your goal now is to count the number of times the sum of measurements in this sliding window increases from the previous sum. So, compare A with B, then compare B with C, then C with D, and so on. Stop when there aren't enough measurements left to create a new three-measurement sum.
|
||||
///
|
||||
/// In the above example, the sum of each three-measurement window is as follows:
|
||||
///
|
||||
/// ```text
|
||||
/// A: 607 (N/A - no previous sum)
|
||||
/// B: 618 (increased)
|
||||
/// C: 618 (no change)
|
||||
/// D: 617 (decreased)
|
||||
/// E: 647 (increased)
|
||||
/// F: 716 (increased)
|
||||
/// G: 769 (increased)
|
||||
/// H: 792 (increased)
|
||||
/// ```
|
||||
///
|
||||
/// In this example, there are 5 sums that are larger than the previous sum.
|
||||
///
|
||||
/// Consider sums of a three-measurement sliding window. How many sums are larger than the previous sum?
|
||||
///
|
||||
pub fn part2(input: &str) -> u32 {
|
||||
let input = input_parse(input);
|
||||
input.iter()
|
||||
.tuple_windows::<(_,_,_)>()
|
||||
.map(|(a,b,c) | a + b + c)
|
||||
.tuple_windows::<(_,_)>()
|
||||
.fold(0, |acc, inp| if inp.0 < inp.1 { acc + 1 } else { acc } )
|
||||
}
|
52
src/main.rs
Normal file
52
src/main.rs
Normal file
|
@ -0,0 +1,52 @@
|
|||
extern crate clap;
|
||||
|
||||
use std::error::Error;
|
||||
use std::fs::File;
|
||||
use std::io::{BufReader, Read};
|
||||
use clap::{arg_enum, clap_app, value_t};
|
||||
|
||||
arg_enum! {
|
||||
#[derive(Debug)]
|
||||
enum DaysImplemented {
|
||||
Day1,
|
||||
}
|
||||
}
|
||||
|
||||
fn main() -> Result<(), Box<dyn Error>> {
|
||||
let matches = clap_app!(AdventOfCode2021 =>
|
||||
(author: "Anthony Cicchetti")
|
||||
(@setting ArgRequiredElseHelp)
|
||||
(@arg day_to_execute: -d --day * +takes_value possible_values(&DaysImplemented::variants()))
|
||||
(@arg part: -p --part * +takes_value possible_values(&["1", "2"]))
|
||||
(@arg input_file: -f --file {file_exists} * conflicts_with[input_string] +takes_value "Input file")
|
||||
(@arg input_string: required_unless[input_file] "Input string")
|
||||
).get_matches();
|
||||
let day: DaysImplemented =
|
||||
value_t!(matches.value_of("day_to_execute"), DaysImplemented).unwrap(); // Safe unwrap because value is required
|
||||
let part = value_t!(matches.value_of("part"), u8).unwrap();
|
||||
let input = match matches.value_of("input_file") {
|
||||
None => value_t!(matches.value_of("input_string"), String).unwrap(),
|
||||
Some(path) => {
|
||||
let mut s = String::new();
|
||||
let _ = BufReader::new(File::open(path)?).read_to_string(&mut s);
|
||||
s
|
||||
}
|
||||
};
|
||||
|
||||
println!("Executing {}, Part {}", day, part);
|
||||
match (day, part) {
|
||||
(DaysImplemented::Day1, 1) => println!("{}", day_01::part1(&input)),
|
||||
(DaysImplemented::Day1, 2) => println!("{}", day_01::part2(&input)),
|
||||
_ => {unimplemented!()}
|
||||
};
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn file_exists(val: String) -> Result<(), String> {
|
||||
if std::fs::metadata(&val).is_ok() {
|
||||
Ok(())
|
||||
} else {
|
||||
Err(String::from(format!("File at {} doesn't exist", &val)))
|
||||
}
|
||||
}
|
Loading…
Add table
Reference in a new issue