01020304050607080910111213141516171819202122232425

Advent of Code

2022/16

Proboscidea Volcanium

in C#

by encse

The sensors have led you to the origin of the distress signal: yet another handheld device, just like the one the Elves gave you. However, you don't see any Elves around; instead, the device is surrounded by elephants! They must have gotten lost in these tunnels, and one of the elephants apparently figured out how to turn on the distress signal.

The ground rumbles again, much stronger this time. What kind of cave is this, exactly? You scan the cave with your handheld device; it reports mostly igneous rock, some ash, pockets of pressurized gas, magma... this isn't just a cave, it's a volcano!

Read the full puzzle.

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text.RegularExpressions;

namespace AdventOfCode.Y2022.Day16;

[ProblemName("Proboscidea Volcanium")]
class Solution : Solver {

    record Map(int[,] distances, Valve[] valves);
    record Valve(int id, string name, int flowRate, string[] tunnels);

    public object PartOne(string input) {
        return Solve(input, true, 30);
    }
    public object PartTwo(string input) {
        return Solve(input, false, 26);
    }

    int Solve(string input, bool humanOnly, int time) {
        var map = Parse(input);
        var start = map.valves.Single(x => x.name == "AA");
        var valvesToOpen = map.valves.Where(valve => valve.flowRate > 0).ToArray();

        var cache = new Dictionary<string, int>();
        if (humanOnly) {
            return MaxFlow(cache, map, start, valvesToOpen.ToHashSet(), time);
        } else {
            return Pairings(valvesToOpen).Select(pairing =>
                 MaxFlow(cache, map, start, pairing.human, time) +
                 MaxFlow(cache, map, start, pairing.elephant, time)
            ).Max();
        }
    }

    // Divide the valves between human and elephant in all possible ways
    IEnumerable<(HashSet<Valve> human, HashSet<Valve> elephant)> Pairings(Valve[] valves) {
        var maxMask = 1 << (valves.Length - 1);

        for (var mask = 0; mask < maxMask; mask++) {
            var elephant = new HashSet<Valve>();
            var human = new HashSet<Valve>();

            elephant.Add(valves[0]);

            for (var ivalve = 1; ivalve < valves.Length; ivalve++) {
                if ((mask & (1 << ivalve)) == 0) {
                    human.Add(valves[ivalve]);
                } else {
                    elephant.Add(valves[ivalve]);
                }
            }
            yield return (human, elephant);
        }
    }

    int MaxFlow(
        Dictionary<string, int> cache,
        Map map,
        Valve currentValve,
        HashSet<Valve> valves,
        int remainingTime
    ) {
        string key =
            remainingTime + "-" +
            currentValve.id + "-" +
            string.Join("-", valves.OrderBy(x => x.id).Select(x => x.id));

        if (!cache.ContainsKey(key)) {
            // current valve gives us this much flow:
            var flowFromValve = currentValve.flowRate * remainingTime;

            // determine best use of the remaining time:
            var flowFromRest = 0;
            foreach (var valve in valves.ToArray()) {
                var distance = map.distances[currentValve.id, valve.id];

                if (remainingTime >= distance + 1) {
                    valves.Remove(valve);
                    remainingTime -= distance + 1;

                    flowFromRest = Math.Max(
                        flowFromRest, MaxFlow(cache, map, valve, valves, remainingTime));

                    remainingTime += distance + 1;
                    valves.Add(valve);
                }

            }
            cache[key] = flowFromValve + flowFromRest;
        }
        return cache[key];
    }

    Map Parse(string input) {
        // Valve BB has flow rate=0; tunnels lead to valve CC
        // Valve CC has flow rate=10; tunnels lead to valves DD, EE
        var valveList = new List<Valve>();
        foreach (var line in input.Split("\n")) {
            var name = Regex.Match(line, "Valve (.*) has").Groups[1].Value;
            var flow = int.Parse(Regex.Match(line, @"\d+").Groups[0].Value);
            var tunnels = Regex.Match(line, "to valves? (.*)").Groups[1].Value.Split(", ");
            valveList.Add(new Valve(0, name, flow, tunnels));
        }
        var valves = valveList
            .OrderByDescending(valve => valve.flowRate)
            .Select((v, i) => v with { id = i })
            .ToArray();

        return new Map(ComputeDistances(valves), valves);
    }

    int[,] ComputeDistances(Valve[] valves) {
        // Floyd-Warshall style distance calculation for every pair of valves.
        // See https://en.wikipedia.org/wiki/Floyd%E2%80%93Warshall_algorithm
        // The only change is that moving from i to i is not enabled.
        
        // This is an O(n^3) algorithm, but we are dealing with a low n.
        var n = valves.Length;

        // Just "big enough" so that infinity + infinity still fits in an int.
        var infinity = int.MaxValue / 2; 

        var dist = new int[valves.Length, valves.Length];
        for (var i = 0; i < n; i++) {
            for (var j = 0; j < n; j++) {
                var neighbours = valves[i].tunnels.Contains(valves[j].name);
                dist[i, j] = neighbours ? 1 : infinity;
            }
        }

        for (var k = 0; k < n; k++) {
            for (var i = 0; i < n; i++) {
                for (var j = 0; j < n; j++) {
                    dist[i, j] = Math.Min(dist[i,j], dist[i, k] + dist[k, j]);
                }
            }
        }
        return dist;
    }
}

Please ☆ my repo if you like it!

© 2025 Advent of Code is a registered trademark in the US Images provided by Bing image creator