--- Day 7: Bridge Repair ---
The Historians take you to a familiar rope bridge over a river in the middle of a jungle. The Chief isn't on this side of the bridge, though; maybe he's on the other side?
When you go to cross the bridge, you notice a group of engineers trying to repair it. (Apparently, it breaks pretty frequently.) You won't be able to cross until it's fixed.
You ask how long it'll take; the engineers tell you that it only needs final calibrations, but some young elephants were playing nearby and stole all the operators from their calibration equations! They could finish the calibrations if only someone could determine which test values could possibly be produced by placing any combination of operators into their calibration equations (your puzzle input).
For example:
190: 10 19 3267: 81 40 27 83: 17 5 156: 15 6 7290: 6 8 6 15 161011: 16 10 13 192: 17 8 14 21037: 9 7 18 13 292: 11 6 16 20
Each line represents a single equation. The test value appears before the colon on each line; it is your job to determine whether the remaining numbers can be combined with operators to produce the test value.
Operators are always evaluated left-to-right, not according to precedence rules. Furthermore, numbers in the equations cannot be rearranged. Glancing into the jungle, you can see elephants holding two different types of operators: add (
+
) and multiply (*
).Only three of the above equations can be made true by inserting operators:
190: 10 19
has only one position that accepts an operator: between10
and19
. Choosing+
would give29
, but choosing*
would give the test value (10 * 19 = 190
).3267: 81 40 27
has two positions for operators. Of the four possible configurations of the operators, two cause the right side to match the test value:81 + 40 * 27
and81 * 40 + 27
both equal3267
(when evaluated left-to-right)!292: 11 6 16 20
can be solved in exactly one way:11 + 6 * 16 + 20
.The engineers just need the total calibration result, which is the sum of the test values from just the equations that could possibly be true. In the above example, the sum of the test values for the three equations listed above is
3749
.Determine which equations could possibly be true. What is their total calibration result?
show (v; e): get each'("**"; ":") 0: `7.txt
Now, for each value and equation, we generate all possible values using + or *, and then check if the desired value is possible.
Define a variable ops, which will contain the valid operations.
ops:(+;*)
For a given equation, we can use over to build up a list of values.
f over (a;b;c;d)
will be f[f[f[a;b]; c]; d]
So we need a function that takes a list of values as x, and a new value as y, and returns all possible operations.
This can be done with {raze ops .\: (x;y)}
. The .\:
is "apply each left", which uses each item of o to apply to x and y.
The raze is then done to join the multiple lists together.
This can be seen in action below:
{raze 0N!ops .\: (x;y)}[10 20 30; 5] (15 25 35;50 100 150) 15 25 35 50 100 150
Now, use over
to generate all possible values.
{raze ops .\: (x;y)} over 3 4 5 12 17 35 60
We can see the results of 3+4+5
, 3*4+5
, 3+4*5
, and 3*4*5
.
Note that these are all evaluated left to right.
Putting it all together, part 1 can be solved as:
sum v{x*x in {raze ops .\:(x;y)}over y}'e
--- Part Two ---
The engineers seem concerned; the total calibration result you gave them is nowhere close to being within safety tolerances. Just then, you spot your mistake: some well-hidden elephants are holding a third type of operator.
The concatenation operator (
||
) combines the digits from its left and right inputs into a single number. For example,12 || 345
would become12345
. All operators are still evaluated left-to-right.Now, apart from the three equations that could be made true using only addition and multiplication, the above example has three more equations that can be made true by inserting operators:
156: 15 6
can be made true through a single concatenation:15 || 6 = 156
.7290: 6 8 6 15
can be made true using6 * 8 || 6 * 15
.192: 17 8 14
can be made true using17 || 8 + 14
.Adding up all six test values (the three that could be made before using only
+
and*
plus the new three that can now be made by also using||
) produces the new total calibration result of11387
.Using your new knowledge of elephant hiding spots, determine which equations could possibly be true. What is their total calibration result?
Now we have a new operator, concatenation.
Concatenation is equivalent to multiplying the left number by a power of 10, and then adding the right one.
Which power of 10? That depends on the number of digits of y.
{y+x*/(count string y)#10}[12 345 6789;45] 1245 34545 678945
We can see this gives the correct results of concatenating 45 on to each of those values.
Now, redefine ops to include this new operation, and run the code from part 1 again.
ops:(+;*;{y+x*/(count string y)#10})
sum v{x*x in {raze ops .\:(x;y)}over y}'e 11387
(v; e): get each'("**"; ":") 0: `:i/7.txt
ops: (+;*)
sum v{x*x in {raze ops .\:(x;y)}over y}'e
ops: (+;*;{y+x*/(count string y)#10})
sum v{x*x in {raze ops .\:(x;y)}over y}'e
Of course, repeating the lines is possibly the easiest solution, but not the nicest looking.
We could add an additional argument to the function that selects how many of the operations to use.
(v; e): get each'("**"; ":") 0: `:i/7.txt
ops: (+;*;{y+x*/(count string y)#10})
{sum v{x*x in {raze (z#ops) .\:(x;y)}[;;z] over y}[;;x]'e} each 2 3
Having to pass z through twice is slightly ugly, but doing part 1 and 2 at the same time is nice.
3 xexp count list
items.
All of these operations can only increase the values, so filtering out ones greater than our target would be an easy start.
We could also do it "backwards", only using multiplication or concatenation where we know it's possible.
For now, this solution runs nearly instantly on the real input, so this additional optimisation doesn't matter too much.