Add 2020/13

This commit is contained in:
kageru 2020-12-13 11:45:17 +01:00
parent da5a57089c
commit e2464b1d3e
Signed by: kageru
GPG Key ID: 8282A2BEA4ADA3D2
2 changed files with 84 additions and 0 deletions

2
2020/inputs/day13 Normal file
View File

@ -0,0 +1,2 @@
1000053
19,x,x,x,x,x,x,x,x,x,x,x,x,37,x,x,x,x,x,523,x,x,x,x,x,x,x,x,x,x,x,x,x,x,x,x,x,13,x,x,x,x,23,x,x,x,x,x,29,x,547,x,x,x,x,x,x,x,x,x,41,x,x,x,x,x,x,17

82
2020/src/bin/day13.rs Normal file
View File

@ -0,0 +1,82 @@
#![feature(test, destructuring_assignment)]
extern crate test;
use aoc2020::common::*;
use itertools::Itertools;
// Port of the rosetta code Python implementation
fn chinese_remainder(divs: Vec<i64>, rems: Vec<i64>) -> i64 {
fn mul_inv(mut a: i64, mut b: i64) -> i64 {
if b == 1 {
return 1;
}
let (mut x0, mut x1, b0) = (0, 1, b);
while a > 1 {
let q = a / b;
(a, b) = (b, a % b);
(x0, x1) = (x1 - q * x0, x0);
}
if x1 < 0 {
x1 += b0;
}
return x1;
}
let mut sum = 0;
let prod: i64 = divs.iter().product();
for (div, rem) in divs.iter().zip(rems.iter()) {
let p = prod / div;
sum += rem * mul_inv(p, *div) * p;
}
return sum % prod;
}
type Parsed = (i64, Vec<Option<i64>>);
fn read_input() -> String {
read_file(13)
}
fn parse_input(raw: &str) -> Parsed {
let (first, second) = raw.lines().next_tuple().unwrap();
(first.parse().unwrap(), second.split(',').map(|n| n.parse().ok()).collect())
}
fn part1((start, nums): &Parsed) -> i64 {
let nums = nums.iter().filter_map(|&n| n).collect_vec();
let (eta, line) = (*start..).find_map(|t| nums.iter().find(|&n| t % n == 0).map(|n| (t, n))).unwrap();
(eta - start) * line
}
fn part2((_, lines): &Parsed) -> i64 {
let rems = lines.iter().enumerate().filter_map(|(n, l)| l.map(|l| l - n as i64)).collect_vec();
let lines = lines.iter().filter_map(|&l| l).collect_vec();
chinese_remainder(lines, rems)
}
fn main() {
let input = parse_input(&read_input());
println!("Part 1: {}", part1(&input));
println!("Part 2: {}", part2(&input));
}
#[cfg(test)]
mod tests {
use super::*;
use aoc2020::*;
use paste::paste;
use test::black_box;
const TEST_INPUT: &str = "939
7,13,x,x,59,x,31,19";
#[test]
fn chinese_remainder_test() {
let divs = vec![3, 5, 7];
let rems = vec![2, 3, 2];
assert_eq!(chinese_remainder(divs, rems), 23);
}
test!(part1() == 295);
test!(part2() == 1068781);
bench!(part1() == 102);
bench!(part2() == 327300950120029);
}