Solve 10
This commit is contained in:
		
							parent
							
								
									e7b127ea01
								
							
						
					
					
						commit
						79b809c3b7
					
				
					 1 changed files with 111 additions and 0 deletions
				
			
		
							
								
								
									
										111
									
								
								src/bin/10.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										111
									
								
								src/bin/10.rs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,111 @@ | |||
| use aoc::*; | ||||
| use glam::IVec2; | ||||
| use std::collections::{HashMap, HashSet}; | ||||
| use std::iter; | ||||
| 
 | ||||
| const INPUT: &str = include_str!("../../input/10"); | ||||
| 
 | ||||
| fn main() { | ||||
|     assert_example!(part1, "10-test", 36); | ||||
|     println!("Part 1: {}", part1(INPUT)); | ||||
|     assert_example!(part2, "10-test", 81); | ||||
|     println!("Part 2: {}", part2(INPUT)); | ||||
| } | ||||
| 
 | ||||
| fn part1(input: &str) -> usize { | ||||
|     let map = Map::parse(input); | ||||
|     map.trailheads().map(|t| map.reachable(t).len()).sum() | ||||
| } | ||||
| 
 | ||||
| fn part2(input: &str) -> usize { | ||||
|     let map = Map::parse(input); | ||||
|     map.trailheads().map(|t| map.distinct_trails(t)).sum() | ||||
| } | ||||
| 
 | ||||
| fn directions() -> [IVec2; 4] { | ||||
|     [ | ||||
|         IVec2::new(1, 0), | ||||
|         IVec2::new(-1, 0), | ||||
|         IVec2::new(0, 1), | ||||
|         IVec2::new(0, -1), | ||||
|     ] | ||||
| } | ||||
| 
 | ||||
| #[derive(Debug)] | ||||
| struct Map { | ||||
|     height: HashMap<IVec2, i64>, | ||||
| } | ||||
| 
 | ||||
| impl Map { | ||||
|     fn reachable(&self, from: IVec2) -> HashSet<IVec2> { | ||||
|         let from_height = match self.get_height(from) { | ||||
|             Some(9) => return [from].into(), | ||||
|             None => return [].into(), | ||||
|             Some(h) => h, | ||||
|         }; | ||||
| 
 | ||||
|         self.climb_up(from, from_height) | ||||
|             .into_iter() | ||||
|             .map(|pos| self.reachable(pos)) | ||||
|             .fold(HashSet::new(), |mut acc, other| { | ||||
|                 acc.extend(other); | ||||
|                 acc | ||||
|             }) | ||||
|     } | ||||
| 
 | ||||
|     fn distinct_trails(&self, from: IVec2) -> usize { | ||||
|         let from_height = match self.get_height(from) { | ||||
|             Some(9) => return 1, | ||||
|             None => return 0, | ||||
|             Some(h) => h, | ||||
|         }; | ||||
| 
 | ||||
|         self.climb_up(from, from_height) | ||||
|             .into_iter() | ||||
|             .map(|pos| self.distinct_trails(pos)) | ||||
|             .sum() | ||||
|     } | ||||
| 
 | ||||
|     fn climb_up(&self, from: IVec2, from_height: i64) -> Vec<IVec2> { | ||||
|         directions() | ||||
|             .into_iter() | ||||
|             .filter_map(|dir| { | ||||
|                 let pos = from + dir; | ||||
|                 self.get_height(pos).map(|height| (pos, height)) | ||||
|             }) | ||||
|             .filter_map(|(pos, h)| { | ||||
|                 if h == from_height + 1 { | ||||
|                     Some(pos) | ||||
|                 } else { | ||||
|                     None | ||||
|                 } | ||||
|             }) | ||||
|             .collect() | ||||
|     } | ||||
| 
 | ||||
|     fn get_height(&self, at: IVec2) -> Option<i64> { | ||||
|         self.height.get(&at).copied() | ||||
|     } | ||||
| 
 | ||||
|     fn trailheads(&self) -> impl Iterator<Item = IVec2> + use<'_> { | ||||
|         self.height.iter().filter_map(|(&pos, &h)| match h { | ||||
|             0 => Some(pos), | ||||
|             _ => None, | ||||
|         }) | ||||
|     } | ||||
| 
 | ||||
|     fn parse(input: &str) -> Self { | ||||
|         let height = input | ||||
|             .lines() | ||||
|             .enumerate() | ||||
|             .flat_map(|(y, line)| line.chars().enumerate().zip(iter::repeat(y))) | ||||
|             .map(|((x, c), y)| { | ||||
|                 let pos = IVec2::new(x as i32, y as i32); | ||||
|                 let height = c.to_digit(10).unwrap() as i64; | ||||
|                 (pos, height) | ||||
|             }) | ||||
|             .collect(); | ||||
| 
 | ||||
|         Self { height } | ||||
|     } | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue