-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDay09.hs
87 lines (78 loc) · 2.24 KB
/
Day09.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
-- |
-- Module : AOC2020.Day09
-- License : BSD3
--
-- Stability : experimental
-- Portability : non-portable
--
-- Day 9. See "AOC.Solver" for the types used in this module!
module AOC2020.Day09 (
day09a,
day09b,
)
where
import AOC.Common (firstJust, slidingWindows)
import AOC.Solver (dyno_, (:~>) (..))
import Control.Monad (guard)
import Data.Foldable (toList)
import Data.List (scanl', tails)
import Data.Sequence (Seq (..))
import qualified Data.Vector as V
import Text.Read (readMaybe)
isBad :: Seq Int -> Maybe Int
isBad xs0 = do
(xs :|> x) <- pure xs0
let badCheck = null do
y : ys <- tails (toList xs)
z <- ys
guard $ (y + z) == x
x <$ guard badCheck
oddOneOut :: Int -> [Int] -> Maybe Int
oddOneOut w = firstJust isBad . slidingWindows (w + 1)
day09a :: [Int] :~> Int
day09a =
MkSol
{ sParse = traverse readMaybe . lines
, sShow = show
, sSolve = oddOneOut (dyno_ "window" 25)
}
findBounds :: V.Vector Int -> Int -> Maybe (Int, Int)
findBounds ns goal = go 0 1
where
go !i !j = do
x <- ns V.!? i
y <- ns V.!? j
case compare (y - x) goal of
LT -> go i (j + 1)
EQ -> pure (i, j)
GT -> go (i + 1) j
day09b :: [Int] :~> (Int, Int)
day09b =
MkSol
{ sParse = traverse readMaybe . lines
, sShow = \(x, y) -> show (x + y)
, sSolve = \ns -> do
goal <- oddOneOut (dyno_ "window" 25) ns
let cumsum = V.fromList (scanl' (+) 0 ns)
(i, j) <- findBounds cumsum goal
let xs = take (j - i) . drop i $ ns
pure (minimum xs, maximum xs)
}
-- an implementation using a priority search queue, which should have
-- efficient lookup and popping. but unfortunately it has too much overhead
-- to offer any overall advantage
-- isBad2 :: IntPSQ Int () -> Maybe Int
-- isBad2 q = do
-- (goal, _, _, xs) <- IntPSQ.minView q
-- let badCheck = null do
-- (x,_,_) <- IntPSQ.toList xs
-- let y = goal - x
-- guard $ y > x
-- guard $ y `IntPSQ.member` xs
-- goal <$ guard badCheck
-- oddOneOut2 :: Int -> [Int] -> Maybe Int
-- oddOneOut2 w = firstJust isBad2
-- . reverse
-- . sortedSlidingWindowsInt (w + 1)
-- . reverse
-- . map (,())