advent-of-code-2020/day7/day7.cpp

194 lines
5.3 KiB
C++

#include <folly/String.h>
#include <algorithm>
#include <execution>
#include <fstream>
#include <iostream>
#include <map>
#include <numeric>
#include <regex>
#include <set>
#include <string_view>
#include <vector>
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::string, std::vector<ContainedBag>>;
std::vector<std::string> readInput(const std::string &filename) {
std::vector<std::string> 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<std::string, std::vector<ContainedBag>> A bag, and the bags it contains
*/
std::pair<std::string, std::vector<ContainedBag>> 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<std::string> unparsedBags;
folly::split(BAG_DELIM, allUnparsedBags, unparsedBags);
std::vector<ContainedBag> 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<std::string, std::vector<ContainedBag>>(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<std::string> &input) {
std::map<std::string, std::vector<ContainedBag>> 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<std::string> 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<std::string> &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<std::string> &input) {
auto bagMap = makeBagMap(input);
int count = 0;
std::stack<std::string> 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<ContainedBag> childBags = bagMap.at(toVisit);
return std::accumulate(childBags.cbegin(), childBags.cend(), 0, [&bagMap](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] << " <input_file>" << 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;
}