|
| 1 | +#include <algorithm> |
| 2 | +#include <fstream> |
| 3 | +#include <functional> |
| 4 | +#include <iostream> |
| 5 | +#include <queue> |
| 6 | +#include <string> |
| 7 | +#include <vector> |
| 8 | + |
| 9 | +std::tuple<long long, long long> monkey_behaviour_model(const long long& old_level, const long long& rationalizer, const long long& to_multiply, const long long& to_add, const long long& to_divide, const long long& m1, const long long& m2) { |
| 10 | + const auto new_level = (old_level * to_multiply + to_add) % rationalizer; |
| 11 | + return {new_level, new_level % to_divide == 0 ? m1 : m2 }; |
| 12 | +} |
| 13 | + |
| 14 | +struct Monkey { |
| 15 | + Monkey(const long long& id, const std::queue<long long>& item_worry_levels, const std::function<std::tuple<long long, long long>(long long, long long)>& action) |
| 16 | + : id(id), item_worry_levels(item_worry_levels), action(action) {} |
| 17 | + |
| 18 | + long long id; |
| 19 | + long long count = 0; |
| 20 | + std::queue<long long> item_worry_levels; |
| 21 | + std::function<std::tuple<long long, long long>(long long, long long)> action; |
| 22 | + |
| 23 | + void print() { |
| 24 | + std::cout << "Monkey " << id << ": "; |
| 25 | + const auto s = item_worry_levels.size(); |
| 26 | + for (long long i = 0; i < s; i++) { |
| 27 | + const auto ele = item_worry_levels.front(); |
| 28 | + item_worry_levels.pop(); |
| 29 | + std::cout << ele << ' '; |
| 30 | + item_worry_levels.push(ele); |
| 31 | + } |
| 32 | + std::cout << '\n'; |
| 33 | + } |
| 34 | +}; |
| 35 | + |
| 36 | +std::function<std::tuple<long long, long long>(long long, long long)> get_monkey_behaviour(const std::vector<std::string>& args) { |
| 37 | + // Args consists of the following values: |
| 38 | + // to_multiply, to_add, to_divide, monkey id if test() is true, monkey id if test() is false |
| 39 | + if (args[0] == "" && args[1] == "old") { |
| 40 | + return std::bind(monkey_behaviour_model, std::placeholders::_1, std::placeholders::_2, 1, std::placeholders::_1, std::stoll(args[2]), std::stoll(args[3]), std::stoll(args[4])); |
| 41 | + } |
| 42 | + else if (args[0] == "old" && args[1] == "") { |
| 43 | + return std::bind(monkey_behaviour_model, std::placeholders::_1, std::placeholders::_2, std::placeholders::_1, 0, std::stoll(args[2]), std::stoll(args[3]), std::stoll(args[4])); |
| 44 | + } |
| 45 | + else if (args[0] == "") { |
| 46 | + return std::bind(monkey_behaviour_model, std::placeholders::_1, std::placeholders::_2, 1, std::stoll(args[1]), std::stoll(args[2]), std::stoll(args[3]), std::stoll(args[4])); |
| 47 | + } |
| 48 | + else if (args[1] == "") { |
| 49 | + return std::bind(monkey_behaviour_model, std::placeholders::_1, std::placeholders::_2, std::stoll(args[0]), 0, std::stoll(args[2]), std::stoll(args[3]), std::stoll(args[4])); |
| 50 | + } |
| 51 | + else { |
| 52 | + std::cout << "This should not happen" << '\n'; |
| 53 | + exit(0); |
| 54 | + } |
| 55 | +} |
| 56 | + |
| 57 | +// Assumes queue has at least 1 item; can be esily modified to generalize |
| 58 | +std::queue<long long> parse_starting_items(const std::string& line) { |
| 59 | + std::queue<long long> q; |
| 60 | + size_t start = 17; |
| 61 | + size_t end = line.find(", ", start); |
| 62 | + while (end != std::string::npos) { |
| 63 | + q.push(std::stoll(line.substr(start, end - start))); |
| 64 | + start = end + 2; |
| 65 | + end = line.find(", ", start); |
| 66 | + } |
| 67 | + q.push(std::stoll(line.substr(start, line.size() - start))); |
| 68 | + return q; |
| 69 | +} |
| 70 | + |
| 71 | +int main(int argc, char * argv[]) { |
| 72 | + std::string input = "../input/day_11_input"; |
| 73 | + if (argc > 1) { |
| 74 | + input = argv[1]; |
| 75 | + } |
| 76 | + |
| 77 | + std::string line; |
| 78 | + std::fstream file(input); |
| 79 | + |
| 80 | + std::vector<Monkey> monkeys; |
| 81 | + |
| 82 | + long long rationalizer = 1; |
| 83 | + while(std::getline(file, line)) { |
| 84 | + const long long id = std::stoll(line.substr(7, line.size() - 7)); |
| 85 | + std::getline(file, line); |
| 86 | + std::queue<long long> item_worry_levels = parse_starting_items(line); |
| 87 | + |
| 88 | + std::vector<std::string> args; |
| 89 | + // Args consists of the following values: |
| 90 | + // to_multiply, to_add, to_divide, monkey id if test() is true, monkey id if test() is false |
| 91 | + |
| 92 | + std::getline(file, line); // Get sign & value used in operation |
| 93 | + if (line[23] == '+') { |
| 94 | + args.emplace_back(); |
| 95 | + args.push_back(line.substr(25, line.size() - 25)); |
| 96 | + } |
| 97 | + else if (line[23] == '*') { |
| 98 | + args.push_back(line.substr(25, line.size() - 25)); |
| 99 | + args.emplace_back(); |
| 100 | + } else { |
| 101 | + std::cout << "This should not happen" << '\n'; |
| 102 | + exit(0); |
| 103 | + } |
| 104 | + std::getline(file, line); // Get value used in test |
| 105 | + const auto val_used_in_test_s = line.substr(21, line.size() - 21); |
| 106 | + args.push_back(val_used_in_test_s); |
| 107 | + const auto val_used_in_test_ll = std::stoll(val_used_in_test_s); |
| 108 | + if (rationalizer % val_used_in_test_ll != 0) { |
| 109 | + rationalizer *= val_used_in_test_ll; |
| 110 | + } |
| 111 | + std::getline(file, line); // Get id of monkey when test is true |
| 112 | + args.push_back(line.substr(29, line.size() - 29)); |
| 113 | + std::getline(file, line); // Get id of monkey when test is false |
| 114 | + args.push_back(line.substr(30, line.size() - 30)); |
| 115 | + |
| 116 | + monkeys.emplace_back(id, item_worry_levels, get_monkey_behaviour(args)); |
| 117 | + std::getline(file, line); |
| 118 | + } |
| 119 | + |
| 120 | + // Assumes monkey id and monkey index is the same; can be esily modified to generalize |
| 121 | + constexpr long long n_rounds = 10000; |
| 122 | + for (long long round = 0; round < n_rounds; round++) { |
| 123 | + for (long long monkey_index = 0; monkey_index < monkeys.size(); monkey_index++) { |
| 124 | + while (!monkeys[monkey_index].item_worry_levels.empty()) { |
| 125 | + monkeys[monkey_index].count++; |
| 126 | + const auto worry = monkeys[monkey_index].item_worry_levels.front(); |
| 127 | + monkeys[monkey_index].item_worry_levels.pop(); |
| 128 | + const auto [new_worry, new_monkey_index] = monkeys[monkey_index].action(worry, rationalizer); |
| 129 | + // std::cout << "Moving " << worry << " from " << monkey << " as " << new_worry << " to " << new_monkey << '\n'; |
| 130 | + monkeys[new_monkey_index].item_worry_levels.push(new_worry); |
| 131 | + } |
| 132 | + } |
| 133 | + } |
| 134 | + |
| 135 | + // for (auto& monkey : monkeys) { |
| 136 | + // monkey.print(); |
| 137 | + // } |
| 138 | + |
| 139 | + std::nth_element(std::begin(monkeys), std::next(std::begin(monkeys), 2), std::end(monkeys), [](const auto& m1, const auto& m2) {return m1.count > m2.count;}); |
| 140 | + std::cout << monkeys[0].count * monkeys[1].count << '\n'; |
| 141 | + // std::cout << rationalizer << '\n'; |
| 142 | + return 0; |
| 143 | +} |
0 commit comments