Today, The Historians take you up to the hot springs on Gear Island! Very suspiciously, absolutely nothing goes wrong as they begin their careful search of the vast field of helixes.
Could this finally be your chance to visit the onsen next door? Only one way to find out.
After a brief conversation with the reception staff at the onsen front desk, you discover that you don't have the right kind of money to pay the admission fee. However, before you can leave, the staff get your attention. Apparently, they've heard about how you helped at the hot springs, and they're willing to make a deal: if you can simply help them arrange their towels, they'll let you in for free!
Every towel at this onsen is marked with a pattern of colored stripes. There are only a few patterns, but for any particular pattern, the staff can get you as many towels with that pattern as you need. Each stripe can be white (w
), blue (u
), black (b
), red (r
), or green (g
). So, a towel with the pattern ggr
would have a green stripe, a green stripe, and then a red stripe, in that order. (You can't reverse a pattern by flipping a towel upside-down, as that would cause the onsen logo to face the wrong way.)
The Official Onsen Branding Expert has produced a list of designs - each a long sequence of stripe colors - that they would like to be able to display. You can use any towels you want, but all of the towels' stripes must exactly match the desired design. So, to display the design rgrgr
, you could use two rg
towels and then an r
towel, an rgr
towel and then a gr
towel, or even a single massive rgrgr
towel (assuming such towel patterns were actually available).
To start, collect together all of the available towel patterns and the list of desired designs (your puzzle input). For example:
12345678910r, wr, b, g, bwu, rb, gb, br
brwrr
bggr
gbbr
rrbgbr
ubwu
bwurrg
brgr
bbrgwb
The first line indicates the available towel patterns; in this example, the onsen has unlimited towels with a single red stripe (r
), unlimited towels with a white stripe and then a red stripe (wr
), and so on.
After the blank line, the remaining lines each describe a design the onsen would like to be able to display. In this example, the first design (brwrr
) indicates that the onsen would like to be able to display a black stripe, a red stripe, a white stripe, and then two red stripes, in that order.
Not all designs will be possible with the available towels. In the above example, the designs are possible or impossible as follows:
brwrr
can be made with abr
towel, then awr
towel, and then finally anr
towel.bggr
can be made with ab
towel, twog
towels, and then anr
towel.gbbr
can be made with agb
towel and then abr
towel.rrbgbr
can be made withr
,rb
,g
, andbr
.ubwu
is impossible.bwurrg
can be made withbwu
,r
,r
, andg
.brgr
can be made withbr
,g
, andr
.bbrgwb
is impossible.
In this example, 6
of the eight designs are possible with the available towel patterns.
To get into the onsen as soon as possible, consult your list of towel patterns and desired designs carefully. How many designs are possible?
As long as you cache/memoize the function, this should complete in a reasonable time.
1234567891011121314151617import { memoize } from "lodash";
const canBuildTowel = memoize((towel: string) => {
if (towel === "") return true;
for (const piece of pieces) {
if (towel.startsWith(piece)) {
if (canBuildTowel(towel.slice(piece.length))) {
return true;
}
}
}
return false;
});
The staff don't really like some of the towel arrangements you came up with. To avoid an endless cycle of towel rearrangement, maybe you should just give them every possible option.
Here are all of the different ways the above example's designs can be made:
brwrr
can be made in two different ways: b
, r
, wr
, r
orbr
, wr
, r
.
bggr
can only be made with b
, g
, g
, and r
.
gbbr
can be made 4 different ways:
g
,b
,b
,r
g
,b
,br
gb
,b
,r
gb
,br
rrbgbr
can be made 6 different ways:
r
,r
,b
,g
,b
,r
r
,r
,b
,g
,br
r
,r
,b
,gb
,r
r
,rb
,g
,b
,r
r
,rb
,g
,br
r
,rb
,gb
,r
bwurrg
can only be made with bwu
, r
, r
, and g
.
brgr
can be made in two different ways: b
, r
, g
, r
orbr
, g
, r
.
ubwu
and bbrgwb
are still impossible.
Adding up all of the ways the towels in this example could be arranged into the desired designs yields 16
(2 + 1 + 4 + 6 + 1 + 2
).
They'll let you into the onsen as soon as you have the list. What do you get if you add up the number of different ways you could make each design?
Nearly identical to Part 1, but return 1 if we've found a match instead of true. Keep a running total of matches for each requested towel.
1234567891011121314151617import { memoize } from "lodash";
const canBuildTowel = memoize((towel: string) => {
if (towel === "") return 1;
let count = 0;
for (const piece of pieces) {
if (towel.startsWith(piece)) {
count += canBuildTowel(towel.slice(piece.length));
}
}
return count;
});
Part 1 Time | Part 1 Rank | Part 2 Time | Part 2 Rank |
---|---|---|---|
00:32:43 | 3,789 | 00:36:31 | 2,778 |
This was incredibly easy. The only annoying thing was JavaScript not having a builtin caching mechanism like Python's @cache, so I had to install lodash for it's memoize
function.