{ "cells": [ { "cell_type": "markdown", "metadata": {}, "source": [ "# Bit set counts\n", "\n", "- https://adventofcode.com/2021/day/3\n", "\n", "We are tasked with counting bits in each column of input, to produce two binary numbers. While there are [some great algorithms for bit counting](https://graphics.stanford.edu/~seander/bithacks.html#CountBitsSetNaive), the fact that we are counting bits across columns means we need to use a slightly different approach.\n", "\n", "The input, can be 'processed' by indexing into the column and testing if that column is equal to `'1'`, and summing these booleans (which in Python have integer values `0` and `1`) gives us the number of rows with the bit set. You can then compare that value with the number of lines and set the corresponding bit in gamma and epsilon (with epsilon being the inverse of gamma). By starting at the left-most column, and left-shifting the accumulated gamma and epsilon values, we naturally build up the bits in the correct order.\n" ] }, { "cell_type": "code", "execution_count": 1, "metadata": {}, "outputs": [], "source": [ "def power_consumption(report: list[str]) -> int:\n", " width = len(report[0])\n", " threshold = (len(report) + 1) // 2 # handles odd and even line lengths\n", " gamma = epsilon = 0\n", " for k in range(width):\n", " # The puzzle description just talks about 'most common'; it doesn't\n", " # matter to _my_ puzzle input if we break ties in favour of 1s or 0s.\n", " one_most_common = sum(line[k] == \"1\" for line in report) >= threshold\n", " gamma = gamma << 1 | one_most_common\n", " epsilon = epsilon << 1 | (not one_most_common)\n", " return gamma * epsilon\n", "\n", "\n", "test_report = \"\"\"\\\n", "00100\n", "11110\n", "10110\n", "10111\n", "10101\n", "01111\n", "00111\n", "11100\n", "10000\n", "11001\n", "00010\n", "01010\n", "\"\"\".splitlines()\n", "\n", "assert power_consumption(test_report) == 198" ] }, { "cell_type": "code", "execution_count": 2, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Part 1: 845186\n" ] } ], "source": [ "import aocd\n", "\n", "diagnostic_report = aocd.get_data(day=3, year=2021).splitlines(False)\n", "print(\"Part 1:\", power_consumption(diagnostic_report))" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "# Part 2: Reducing the input numbers based on bit counts\n", "\n", "Instead of accumulating a bit for each column in two numbers, we need to reduce the input lines based on the outcome. We'll have to assume that the puzzle inputs will always lead to a a single line out output for this process.\n", "\n", "For each column index, calculate the threshold, determine if the `'1'` count is high enough, filter the lines based on the column value and and repeat until there is a single line left. Only then do we need to convert from string of binary digits to a number.\n", "\n", "This approach takes $O(n \\log_2 n)$ time as we repeatedly pass over the lines but (roughly) half the number each time.\n" ] }, { "cell_type": "code", "execution_count": 3, "metadata": {}, "outputs": [], "source": [ "def life_support_rating(report: list[str]) -> int:\n", " width = len(report[0])\n", "\n", " def reduce(lines: list[str], filter_on=\"01\") -> int:\n", " for k in range(width):\n", " threshold = (len(lines) + 1) // 2\n", " one_most_common = sum(line[k] == \"1\" for line in lines) >= threshold\n", " lines = [line for line in lines if line[k] == filter_on[one_most_common]]\n", " if len(lines) == 1:\n", " return int(lines[0], 2)\n", "\n", " oxygen_generator_rating = reduce(report)\n", " co2_scrubber_rating = reduce(report, filter_on=\"10\")\n", " return oxygen_generator_rating * co2_scrubber_rating\n", "\n", "\n", "assert life_support_rating(test_report) == 230" ] }, { "cell_type": "code", "execution_count": 4, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Part 2: 4636702\n" ] } ], "source": [ "print(\"Part 2:\", life_support_rating(diagnostic_report))" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "# Optimising part 2 using a binary tree\n", "\n", "We can also solve this challenge using a binary tree.\n", "\n", "Instead of repeatedly looping over the lines to count and then filter, we could build a binary tree structure from the inputs. Each node has left and right subtrees for `0` and `1`, as well as a count of the number of lines that have the same prefix up to this point. Producing the two values is then just a question of following the correct branch down to the leaf, all the while shifting the collected value and including the selected direction via bitwise OR, like in part 1. Here, \"following the correct branch\" means comparing the two counts at that node in the tree and following the higher or lower count to the next node; and what direction you followed determines if a `0` or `1` is added to the accumulating value.\n", "\n", "This approach is also a $O(n \\log_2 n)$ solution; constructing the btree loops over all $n$ lines and then over $\\log2 n$ bits on each line; reading out the numbers is a fixed cost at the end.\n", "\n", "And while I'm fond of using OOP for many of my AoC solutions, the binary tree here can trivially be stored in a list instead; just like\n", "a [the most common implementation of a heap](https://en.wikipedia.org/wiki/Heap_%28data_structure%29#Implementation), a given node for binary value $v_k$ at a given depth $k$ is found at $2^{k} - 2 + v_k$. We'll need a list of length $2^{k + 1} - 2$, and because each cell represents a line count, the values all start at `0`. E.g., given the inputs `001`, `101`, `110` and `111`, you'd create a btree with the following structure:\n", "\n", "\n", "\n", "![]()\n", "\n", "which can be stored in a list starting at the first two nodes under the root at indices 0 and 1, the 2 left hand-side nodes at the next level (representing lines that start with `0`) in indices 2 and 3, then 4 and 5 contain the counts for everything starting with `1`, and so on:\n", "\n", "```python\n", "btree = [1, 3, 1, 0, 1, 2, 0, 1, 0, 0, 0, 1, 1, 1]\n", "# 0 1 0 0 1 1 0 0 0 0 1 1 1 1\n", "# . . 0 1 0 1 0 0 1 1 0 0 1 1\n", "# . . . . . . 0 1 0 1 0 1 0 1\n", "```\n", "\n", "I've dropped the `- 2` subtractions to simplify the code, and just live with two extra unused positions in my btree list.\n" ] }, { "cell_type": "code", "execution_count": 5, "metadata": {}, "outputs": [], "source": [ "from operator import ge, lt\n", "\n", "\n", "def life_support_rating_btree(report: list[str]) -> int:\n", " # build a btree in a list (first two elements are left untouched)\n", " # with each node representing the number of lines that have the same\n", " # bit prefix. e.g. `0` and `1` give the btree [0, 0, 0, 1], while\n", " # `01`, `10` and `11` produces `[0, 0, 1, 2, 0, 1, 1, 1]`.\n", " depth = len(report[0])\n", " btree = [0] * (2 << depth)\n", " for line in report:\n", " v, offset = 0, 1\n", " for c in line:\n", " offset <<= 1\n", " v = (v << 1) | (c == \"1\")\n", " btree[offset + v] += 1\n", "\n", " def rating(op=ge):\n", " # Walk the btree to reduce the line count down to 1 by choosing either\n", " # the left (0) or right (1) child node for the given comparison operator\n", " # comparing between the line counts.\n", " v, offset = 0, 1\n", " for _ in range(depth):\n", " offset <<= 1\n", " v <<= 1\n", " idx = offset + v\n", " bit = op(btree[idx + 1], btree[idx]) # #(1) >= #(0) or #(1) < #(0)\n", " if op is lt and btree[idx + bit] == 1:\n", " # We've reduced the tree to a single line, reconstruct the\n", " # remainder by always using >= (instead of < for the co2\n", " # scrubber rating)\n", " op = ge\n", " v += bit\n", "\n", " return v\n", "\n", " oxygen_generator_rating = rating()\n", " co2_scrubber_rating = rating(lt)\n", " return oxygen_generator_rating * co2_scrubber_rating\n", "\n", "\n", "assert life_support_rating_btree(test_report) == 230" ] }, { "cell_type": "code", "execution_count": 6, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Part 2: 4636702\n" ] } ], "source": [ "print(\"Part 2:\", life_support_rating(diagnostic_report))" ] } ], "metadata": { "interpreter": { "hash": "8bb5fd587ebf4d90f905285c44a569046664a8863ee065ff2dd968491b671e06" }, "kernelspec": { "display_name": "Python 3.10.0 64-bit ('adventofcode-mOkh6lsX': pipenv)", "language": "python", "name": "python3" }, "language_info": { "codemirror_mode": { "name": "ipython", "version": 3 }, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.12.0" }, "orig_nbformat": 4 }, "nbformat": 4, "nbformat_minor": 2 }