#include #include #include #include #include #include #include #include #include #include #include auto constexpr LINE_PATTERN = R"((.*) bags? contain (.*)\.)"; auto constexpr BAG_PATTERN = R"((\d) (.*) bags?)"; auto constexpr NO_OTHER_BAGS = "no other bags"; auto constexpr BAG_DELIM = ", "; auto constexpr DESIRED_BAG = "shiny gold"; /** * Represents a bag this is contained in another bag (i.e. a bag and its quantity) */ class ContainedBag { public: ContainedBag(std::string color, int quantity) : color(color), quantity(quantity) { } const std::string &getColor() const { return this->color; } int getQuantity() const { return this->quantity; } bool operator==(ContainedBag bag) { return bag.getColor() == this->getColor() && bag.getQuantity() == this->getQuantity(); } private: std::string color; int quantity; }; // I would use a set for the value but I don't want to deal with hash functions right now using BagMap = std::map>; std::vector readInput(const std::string &filename) { std::vector input; std::string line; std::ifstream file(filename); while (std::getline(file, line)) { input.push_back(line); } return input; } /** * Parse the input line into a bag and its contents * @param line A line of input * @return std::pair> A bag, and the bags it contains */ std::pair> parseInputLine(const std::string &line) { std::regex lineExpression(LINE_PATTERN); std::smatch lineMatches; if (!std::regex_match(line, lineMatches, lineExpression)) { throw new std::invalid_argument("Invalid input line"); } std::string bagName = lineMatches[1]; std::string allUnparsedBags = lineMatches[2]; std::vector unparsedBags; folly::split(BAG_DELIM, allUnparsedBags, unparsedBags); std::vector containedBags; for (const std::string &unparsedBag : unparsedBags) { // Check for no bags contained if (unparsedBag == NO_OTHER_BAGS) { continue; } // Check for some quantity of bags contained std::regex bagExpression(BAG_PATTERN); std::smatch bagMatches; if (!std::regex_match(unparsedBag, bagMatches, bagExpression)) { throw new std::invalid_argument("Invalid bagspec"); } ContainedBag containedBag(bagMatches[2], std::stoi(bagMatches[1])); containedBags.push_back(containedBag); } return std::pair>(bagName, containedBags); } /** * Make a map of all of the contained bags * @param input The input for the puzzle * @return BagMap A map of bags to their children */ BagMap makeBagMap(const std::vector &input) { std::map> bags; std::transform(input.cbegin(), input.cend(), std::inserter(bags, bags.begin()), [](const std::string &line) { return parseInputLine(line); }); return bags; } /** * Check if one bag contains another * @param bagMap A map of bags to their children * @param originBag The bag to check if it contains desiredBag * @param desiredBag The bag to search for * @return true If the bag is found * @return false False otherwise */ bool doesBagContain(const BagMap &bagMap, const std::string &originBag, const std::string &desiredBag) { std::stack toVisit; toVisit.emplace(originBag); while (!toVisit.empty()) { std::string visiting = toVisit.top(); toVisit.pop(); if (visiting == desiredBag) { return true; } for (ContainedBag bag : bagMap.at(visiting)) { toVisit.emplace(std::string(bag.getColor())); } } return false; } int part1(const std::vector &input) { auto bagMap = makeBagMap(input); int count = 0; for (auto bagEntry : bagMap) { const std::string &bagName = bagEntry.first; if (bagName == DESIRED_BAG) { continue; } count += doesBagContain(bagMap, bagName, DESIRED_BAG); } return count; } int part2(const std::vector &input) { auto bagMap = makeBagMap(input); int count = 0; std::stack toVisit; toVisit.emplace(DESIRED_BAG); while (!toVisit.empty()) { std::string visiting = toVisit.top(); toVisit.pop(); for (ContainedBag bag : bagMap.at(visiting)) { count += bag.getQuantity(); // Enqueue this bagQuantity times. This s slow (exponential blowup), but will work. // Getting rid of this would require making a map of iterating up the whole tree of parents, which isn't fun for (int i = 0; i < bag.getQuantity(); i++) { toVisit.emplace(std::string(bag.getColor())); } } } return count; } int recursivePart2(const BagMap &bagMap, const std::string &toVisit = DESIRED_BAG) { std::vector childBags = bagMap.at(toVisit); return std::accumulate(childBags.cbegin(), childBags.cend(), 0, [&](int total, const ContainedBag &bag) { return total + bag.getQuantity() + bag.getQuantity() * recursivePart2(bagMap, bag.getColor()); }); } int main(int argc, char *argv[]) { if (argc != 2) { std::cerr << argv[0] << " " << std::endl; return 1; } auto input = readInput(argv[1]); std::cout << part1(input) << std::endl; std::cout << part2(input) << std::endl; std::cout << recursivePart2(makeBagMap(input)) << std::endl; }