summaryrefslogtreecommitdiff
path: root/2019/rust/day12/src
diff options
context:
space:
mode:
Diffstat (limited to '2019/rust/day12/src')
-rw-r--r--2019/rust/day12/src/main.rs154
1 files changed, 154 insertions, 0 deletions
diff --git a/2019/rust/day12/src/main.rs b/2019/rust/day12/src/main.rs
new file mode 100644
index 0000000..8653f36
--- /dev/null
+++ b/2019/rust/day12/src/main.rs
@@ -0,0 +1,154 @@
+use num::integer::Integer;
+use regex::Regex;
+use std::collections::HashSet;
+use std::io::{self, BufRead};
+
+type CoordType = i32;
+type EnergyType = i32;
+
+#[derive(Clone, Copy, Debug, Eq, Hash, PartialEq)]
+struct Coords {
+ x: CoordType,
+ y: CoordType,
+ z: CoordType,
+}
+
+#[derive(Clone, Copy, Debug, Eq, Hash, PartialEq)]
+struct Moon {
+ pos: Coords,
+ vel: Coords,
+}
+
+fn read_input() -> Vec<Coords> {
+ let stdin = io::stdin();
+ let re = Regex::new(r"<x=(?P<x>[0-9-]*), *y=(?P<y>[0-9-]*), *z=(?P<z>[0-9-]*)").
+ expect("Regex failed to compile");
+
+ stdin.lock().lines()
+ .map(|line| {
+ match line {
+ Ok(l) => {
+ let caps = match re.captures(&l) {
+ Some(c) => c,
+ _ => panic!("Regex issue for line: {:?}", l),
+ };
+ Coords {
+ x: caps["x"].parse().unwrap(),
+ y: caps["y"].parse().unwrap(),
+ z: caps["z"].parse().unwrap(),
+ }
+ },
+ _ => panic!("Failed to parse line: {:?}", line),
+ }
+ }).collect()
+}
+
+// To apply gravity, consider every pair of moons. On each axis (x, y, and z), the velocity of each
+// moon changes by exactly +1 or -1 to pull the moons together. However, if the positions on a
+// given axis are the same, the velocity on that axis does not change for that pair of moons.
+fn apply_gravity(this: &mut Moon, other: &Moon) {
+ fn vel_change(a: CoordType, b: CoordType) -> CoordType {
+ if a < b { 1 } else if a > b { -1 } else { 0 }
+ }
+
+ this.vel.x += vel_change(this.pos.x, other.pos.x);
+ this.vel.y += vel_change(this.pos.y, other.pos.y);
+ this.vel.z += vel_change(this.pos.z, other.pos.z);
+}
+
+// Once all gravity has been applied, apply velocity: simply add the velocity of each moon to its
+// own position.
+fn apply_velocity(moon: &mut Moon) {
+ moon.pos.x += moon.vel.x;
+ moon.pos.y += moon.vel.y;
+ moon.pos.z += moon.vel.z;
+}
+
+// A moon's potential energy is the sum of the absolute values of its x, y, and z position
+// coordinates.
+fn potential_energy(moon: &Moon) -> EnergyType {
+ moon.pos.x.abs() + moon.pos.y.abs() + moon.pos.z.abs()
+}
+
+// A moon's kinetic energy is the sum of the absolute values of its velocity coordinates. Below,
+// each line shows the calculations for a moon's potential energy (pot), kinetic energy (kin), and
+// total energy:
+fn kinetic_energy(moon: &Moon) -> EnergyType {
+ moon.vel.x.abs() + moon.vel.y.abs() + moon.vel.z.abs()
+}
+
+fn print_moon(moon: &Moon) {
+ println!("pos=<x={:2}, y={:2}, z={:2}>, vel=<x={:2}, y={:2}, z={:2}>",
+ moon.pos.x, moon.pos.y, moon.pos.z,
+ moon.vel.x, moon.vel.y, moon.vel.z,);
+}
+
+fn iterations_until_repeat(mut moons: Vec<Moon>) -> usize{
+ let mut seen = HashSet::new();
+ seen.insert(moons.clone());
+
+ for n in 1.. {
+ let old_moons = moons.clone();
+ for (i, moon) in moons.iter_mut().enumerate() {
+ for (j, old) in old_moons.iter().enumerate() {
+ if i != j { apply_gravity(moon, old) };
+ }
+ }
+ moons.iter_mut().map(|moon| apply_velocity(moon)).last();
+ if ! seen.insert(moons.clone()) {
+ return n;
+ }
+ }
+ 0
+}
+
+fn main() {
+ let coords = read_input();
+ let mut moons: Vec<Moon> = coords.iter()
+ .map(|pos| Moon { pos: *pos, vel: Coords { x: 0, y: 0, z: 0 }}).collect();
+
+ // Simulate the motion of the moons in time steps. Within each time step, first update the
+ // velocity of every moon by applying gravity. Then, once all moons' velocities have been
+ // updated, update the position of every moon by applying velocity. Time progresses by one step
+ // once all of the positions are updated.
+ for _ in 0..std::env::args().nth(1).unwrap().parse::<i128>().unwrap() {
+ let old_moons = moons.clone();
+ for (i, moon) in moons.iter_mut().enumerate() {
+ for (j, old) in old_moons.iter().enumerate() {
+ if i != j { apply_gravity(moon, old) };
+ }
+ }
+ moons.iter_mut().map(|moon| apply_velocity(moon)).last();
+ }
+
+ // Then, it might help to calculate the total energy in the system. The total energy for a
+ // single moon is its potential energy multiplied by its kinetic energy.
+ let energy = moons.iter()
+ .map(|moon| potential_energy(moon) * kinetic_energy(moon)).sum::<EnergyType>();
+
+ // What is the total energy in the system after simulating the moons given in your scan for
+ // 1000 steps?
+ println!("Total energy: {}", energy);
+
+ // Determine the number of steps that must occur before all of the moons' positions and
+ // velocities exactly match a previous point in time.
+ let moons_x: Vec<Moon> = moons.iter().map(|m| Moon {
+ pos: Coords { x: m.pos.x, y: 0, z: 0},
+ vel: Coords { x: m.vel.x, y: 0, z: 0},
+ }).collect();
+ let moons_y: Vec<Moon> = moons.iter().map(|m| Moon {
+ pos: Coords { x: 0, y: m.pos.y, z: 0},
+ vel: Coords { x: 0, y: m.vel.y, z: 0},
+ }).collect();
+ let moons_z: Vec<Moon> = moons.iter().map(|m| Moon {
+ pos: Coords { x: 0, y: 0, z: m.pos.z },
+ vel: Coords { x: 0, y: 0, z: m.vel.z },
+ }).collect();
+ let repeat_x = iterations_until_repeat(moons_x);
+ let repeat_y = iterations_until_repeat(moons_y);
+ let repeat_z = iterations_until_repeat(moons_z);
+ // Clearly, you might need to find a more efficient way to simulate the universe.
+ let repeat = repeat_z.lcm(&repeat_x.lcm(&repeat_y));
+
+ println!("Iterations until repeat: {}", repeat);
+}