Day 3: Mull It Over
Megathread guidelines
- Keep top level comments as only solutions, if you want to say something other than a solution put it in a new post. (replies to comments can be whatever)
- You can send code in code blocks by using three backticks, the code, and then three backticks or use something such as https://topaz.github.io/paste/ if you prefer sending it through a URL
FAQ
- What is this?: Here is a post with a large amount of details: https://programming.dev/post/6637268
- Where do I participate?: https://adventofcode.com/
- Is there a leaderboard for the community?: We have a programming.dev leaderboard with the info on how to join in this post: https://programming.dev/post/6631465
Haskell
Oof, a parsing problem :/ This is some nasty-ass code. step
is almost the State monad written out explicitly.
Solution
import Control.Monad
import Data.Either
import Data.List
import Text.Parsec
data Ins = Mul !Int !Int | Do | Dont
readInput :: String -> [Ins]
readInput = fromRight undefined . parse input ""
where
input = many ins <* many anyChar
ins =
choice . map try $
[ Mul <$> (string "mul(" *> arg) <*> (char ',' *> arg) <* char ')',
Do <$ string "do()",
Dont <$ string "don't()",
anyChar *> ins
]
arg = do
s <- many1 digit
guard $ length s <= 3
return $ read s
run f = snd . foldl' step (True, 0)
where
step (e, a) i =
case i of
Mul x y -> (e, if f e then a + x * y else a)
Do -> (True, a)
Dont -> (False, a)
main = do
input <- readInput <$> readFile "input03"
print $ run (const True) input
print $ run id input
Love to see you chewing through this parsing problem in Haskell, I didn’t dare use Parsec because I wasn’t confident enough.
Why did you decide to have a strict definition of Mul !Int !Int
?
My guess is because a linter and/or HLS was suggesting it. I know HLS used to suggest making your fields strict in almost all cases. In this case I have a hunch that it slightly cuts down on memory usage because we use almost all Mul
s either way. So it does not need to keep the string it is parsed from in memory as part of the thunk.
But it probably makes a small/negligible difference here.