Added AOC 2020 day 03 - 14
This commit is contained in:
		
							
								
								
									
										111
									
								
								www/statics/aoc/2020/07_solution.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										111
									
								
								www/statics/aoc/2020/07_solution.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,111 @@ | ||||
| use crate::common::AdventOfCodeDay; | ||||
|  | ||||
| use std::collections::VecDeque; | ||||
| use std::collections::HashSet; | ||||
| use regex::Regex; | ||||
|  | ||||
| #[derive(Debug)] | ||||
| struct Rule { | ||||
|     parent: String, | ||||
|     children: Vec<(usize, String)>, | ||||
| } | ||||
|  | ||||
| pub struct Day07 { | ||||
|     input: Vec<Rule>, | ||||
| } | ||||
|  | ||||
| fn parse_line(line: &str) -> Rule { | ||||
|     lazy_static! { | ||||
|         static ref REX_LINE:  Regex = Regex::new(r"^(?P<bag>[a-z]+ [a-z]+) bags contain (?P<inner>(?:(?:(?:[0-9]+ (?:[a-z]+ [a-z]+)) bags?(?:, |\.)?)+)|no other bags\.)$").unwrap(); | ||||
|         static ref REX_INNER: Regex = Regex::new(r"(?P<num>[0-9]+) (?P<bag>[a-z]+ [a-z]+) bags?[.,]").unwrap(); | ||||
|     } | ||||
|  | ||||
|     if let Some(rline) = REX_LINE.captures(line) { | ||||
|         let parent = rline.name("bag").unwrap().as_str().to_owned(); | ||||
|  | ||||
|         if rline.name("inner").unwrap().as_str() == "not other bags." { | ||||
|             return Rule { | ||||
|                 parent: parent, | ||||
|                 children: vec![], | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         let childs = REX_INNER | ||||
|                         .captures_iter(rline.name("inner").unwrap().as_str()) | ||||
|                         .map(|p| (p.name("num").unwrap().as_str().parse::<usize>().unwrap(), p.name("bag").unwrap().as_str().to_owned())) | ||||
|                         .collect::<Vec<(usize, String)>>(); | ||||
|  | ||||
|         return Rule { | ||||
|             parent: parent, | ||||
|             children: childs, | ||||
|         } | ||||
|     } else { | ||||
|         panic!(format!("no regex line match: '{}'", line.to_owned())); | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
| impl Day07 { | ||||
|     pub fn new() -> Self { | ||||
|         let input_bytes = include_bytes!("../res/07_input.txt"); | ||||
|         let input_str = String::from_utf8_lossy(input_bytes); | ||||
|          | ||||
|         let data = input_str | ||||
|                         .lines() | ||||
|                         .map(parse_line) | ||||
|                         .collect::<Vec<Rule>>(); | ||||
|  | ||||
|         Self { | ||||
|             input: data | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn find_rule(&self, bag: &str) -> Option<&Rule> { | ||||
|         for r in &self.input { | ||||
|             if r.parent == bag { | ||||
|                 return Some(r); | ||||
|             } | ||||
|         } | ||||
|         return None; | ||||
|     } | ||||
|      | ||||
|     fn can_contain(&self, parent: &str, child: &str) -> bool { | ||||
|         let mut candidates: VecDeque<String> = VecDeque::new(); | ||||
|         let mut visited: HashSet<String> = HashSet::new(); | ||||
|      | ||||
|         candidates.push_back(parent.to_owned()); | ||||
|         visited.insert(parent.to_owned()); | ||||
|      | ||||
|         while !candidates.is_empty() { | ||||
|             let cand = candidates.pop_back().unwrap(); | ||||
|             if let Some(rule) = self.find_rule(cand.as_str()) { | ||||
|                 for rulechild in &rule.children { | ||||
|                     let rcname = rulechild.1.to_owned(); | ||||
|                     if rcname == child { | ||||
|                         return true; | ||||
|                     } else if !visited.contains(&rcname) { | ||||
|                         candidates.push_back(rcname.to_owned()); | ||||
|                         visited.insert(rcname); | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|      | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     fn get_full_children_count(&self, name: &str) -> usize { | ||||
|         self.find_rule(name).unwrap().children.iter().map(|p| p.0 + p.0 * self.get_full_children_count(&p.1)).sum() | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl AdventOfCodeDay for Day07 { | ||||
|  | ||||
|     fn task_1(&self) -> String { | ||||
|         self.input.iter().filter(|p| self.can_contain(p.parent.as_str(), "shiny gold")).count().to_string() | ||||
|     } | ||||
|  | ||||
|     fn task_2(&self) -> String  { | ||||
|         self.get_full_children_count("shiny gold").to_string() | ||||
|     } | ||||
| } | ||||
		Reference in New Issue
	
	Block a user