{-# Language QuasiQuotes, ImportQualifiedPost #-}
module Main where
import Advent
import Data.Map qualified as Map
import Data.Vector.Unboxed (Vector)
import Data.Vector.Unboxed qualified as V
main :: IO ()
IO ()
main =
do input <- [format|2017 6 %u&%t%n|]
print (solve (V.fromList input))
solve :: Vector Int -> (Int,Int)
solve :: Vector Int -> (Int, Int)
solve = [Vector Int] -> (Int, Int)
forall a. Ord a => [a] -> (Int, Int)
findLoop ([Vector Int] -> (Int, Int))
-> (Vector Int -> [Vector Int]) -> Vector Int -> (Int, Int)
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Vector Int -> Vector Int) -> Vector Int -> [Vector Int]
forall a. (a -> a) -> a -> [a]
iterate Vector Int -> Vector Int
step
findLoop :: Ord a => [a] -> (Int,Int)
findLoop :: forall a. Ord a => [a] -> (Int, Int)
findLoop = Map a Int -> [a] -> (Int, Int)
forall {k}. Ord k => Map k Int -> [k] -> (Int, Int)
go Map a Int
forall k a. Map k a
Map.empty
where
go :: Map k Int -> [k] -> (Int, Int)
go Map k Int
seen (k
x:[k]
xs) =
let n :: Int
n = Map k Int -> Int
forall k a. Map k a -> Int
Map.size Map k Int
seen in
case k -> Map k Int -> Maybe Int
forall k a. Ord k => k -> Map k a -> Maybe a
Map.lookup k
x Map k Int
seen of
Maybe Int
Nothing -> Map k Int -> [k] -> (Int, Int)
go (k -> Int -> Map k Int -> Map k Int
forall k a. Ord k => k -> a -> Map k a -> Map k a
Map.insert k
x Int
n Map k Int
seen) [k]
xs
Just Int
i -> (Int
n, Int
nInt -> Int -> Int
forall a. Num a => a -> a -> a
-Int
i)
step :: Vector Int -> Vector Int
step :: Vector Int -> Vector Int
step Vector Int
xs = (Int -> Int -> Int) -> Vector Int -> [(Int, Int)] -> Vector Int
forall a b.
Unbox a =>
(a -> b -> a) -> Vector a -> [(Int, b)] -> Vector a
V.accum Int -> Int -> Int
forall a. Num a => a -> a -> a
(+) Vector Int
xs ((Int
i, -Int
mx) (Int, Int) -> [(Int, Int)] -> [(Int, Int)]
forall a. a -> [a] -> [a]
: [ (Int
jInt -> Int -> Int
forall a. Integral a => a -> a -> a
`rem`Int
n, Int
1) | Int
j <- [Int
iInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1 .. Int
iInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
mx]])
where
mx :: Int
mx = Vector Int -> Int
forall a. (Unbox a, Ord a) => Vector a -> a
V.maximum Vector Int
xs
Just Int
i = Int -> Vector Int -> Maybe Int
forall a. (Unbox a, Eq a) => a -> Vector a -> Maybe Int
V.elemIndex Int
mx Vector Int
xs
n :: Int
n = Vector Int -> Int
forall a. Unbox a => Vector a -> Int
V.length Vector Int
xs