Day 19: Linen Layout
Puzzle Description
Scala Center Link
Scala Center Advent of Code 2024, Day 19
Solution Summary
- Parse input into two
List[String]
-
Calculate each pattern
- For
part1
, we just have to find 1 valid configuration then we can end early. - For
part2
, we have to find all valid configurations and count them.
- For
- Sum results of calculations
Part 1
Part 1 is just finding one valid pattern, which isn't too bad.
Let's start with the parsing:
def parse(str: String): (List[String], List[String]) =
val Array(towelsStr, designStr) = str.split("\n\n")
val towels = towelsStr.split(',').map(_.trim).toList
val designs = designStr.split('\n').toList
(towels.sortBy(_.length)(using Ordering[Int].reverse), designs)
We sort the towels by length here so we don't have to do it in either solution - this doesn't matter too much for part 2, but for part 1 choosing the biggest chunks first is preferable.
Now we just have to find a single valid configuration. My immediate thought was parser combinators, but I didn't really want to pull those out so I just did a recursive function:
def parseDesign(towels: List[String], design: String): Boolean =
def go(curTowels: List[String], restDesign: String): Boolean =
if restDesign.isEmpty then
true
else
curTowels match
case head :: next =>
if restDesign.startsWith(head) then
val nextDesign = restDesign.drop(head.length)
go(towels, nextDesign) || go(next, restDesign)
else
go(next, restDesign)
case Nil => false
go(towels, design)
Part 1 is now just as simple as counting the designs with at least 1 valid configuration:
def part1(str: String): Long =
val (towels, designs) = parse(str)
designs.count(parseDesign(towels, _)).toLong
And that's part 1 solved.
Part 2
It's not too hard to write something that will work in the small example case, but I struggled with making it optimized enough. I was unable to rub my braincells together today so I used merlinorg's (older) solution for part 2.
The key is using a cache to memoize the amount of combinations of a suffix of a pattern. Honestly I don't really fully understand the code myself, so here it is and my best explanation:
def countDesigns(towels: List[String], design: String): Long =
def go(pattern: String, total: Long, cache: Map[String, Long]): (Long, Map[String, Long]) =
cache.get(pattern) match
case Some(count) => (total + count, cache)
case _ =>
val (count, cache2) = towels.foldLeft(0L -> cache):
case ((count, cache), towel) =>
if pattern.startsWith(towel) then
go(pattern.drop(towel.length), count, cache)
else
(count, cache)
(total + count, cache2 + (pattern -> count))
go(design, 0L, Map("" -> 1L))._1
If the pattern is already in the cache, then it returns the cache score + the total arg. If not, then it calculate combinations by going through all the towels, and recursing on ones the pattern starts with. Then, it returns the count calculated from all combinations for that pattern and adds that pattern to the cache.
The cache is useful because when a pattern's start can be matched by multiple towels then the suffixes tend to be equal, and if they aren't they likely will have children later that are equal. So caching means that the bulk of the computation is done at the start but then most of the work is cache fetching.
Now to just hook that up to part2
:
def part2(str: String): Long =
val (towels, designs) = parse(str)
designs.map(countDesigns(towels, _)).sum
Final code:
def parse(str: String): (List[String], List[String]) =
val Array(towelsStr, designStr) = str.split("\n\n")
val towels = towelsStr.split(',').map(_.trim).toList
val designs = designStr.split('\n').toList
(towels.sortBy(_.length)(using Ordering[Int].reverse), designs)
def parseDesign(towels: List[String], design: String): Boolean =
def go(curTowels: List[String], restDesign: String): Boolean =
if restDesign.isEmpty then
true
else
curTowels match
case head :: next =>
if restDesign.startsWith(head) then
val nextDesign = restDesign.drop(head.length)
go(towels, nextDesign) || go(next, restDesign)
else
go(next, restDesign)
case Nil => false
go(towels, design)
def countDesigns(towels: List[String], design: String): Long =
def go(pattern: String, total: Long, cache: Map[String, Long]): (Long, Map[String, Long]) =
cache.get(pattern) match
case Some(count) => (total + count, cache)
case _ =>
val (count, cache2) = towels.foldLeft(0L -> cache):
case ((count, cache), towel) =>
if pattern.startsWith(towel) then
go(pattern.drop(towel.length), count, cache)
else
(count, cache)
(total + count, cache2 + (pattern -> count))
go(design, 0L, Map("" -> 1L))._1
def part1(input: (List[String], List[String])): Long =
val (towels, designs) = input
designs.count(parseDesign(towels, _)).toLong
def part2(input: (List[String], List[String])): Long =
val (towels, designs) = input
designs.map(countDesigns(towels, _)).sum
Benchmark
Part 1
Mean |
Error |
|
---|---|---|
JVM |
11.462 ms |
+/- 0.132 ms |
JS |
11.167 ms |
+/- 0.015 ms |
Native |
10.780 ms |
+/- 0.035 ms |
Part 2
Mean |
Error |
|
---|---|---|
JVM |
64.350 ms |
+/- 1.512 ms |
JS |
100.685 ms |
+/- 0.603 ms |
Native |
191.432 ms |
+/- 0.909 ms |