{-# Language ImportQualifiedPost, QuasiQuotes #-}
module Main (main) where
import Advent
import Advent.Format (format)
import Data.Foldable
import Data.Sequence (Seq(..))
import Data.Sequence qualified as Seq
import Data.Set (Set)
import Data.Set qualified as Set
import Data.Vector.Unboxed (Vector)
import Data.Vector.Unboxed qualified as V
main :: IO ()
IO ()
main =
do (xs,ys) <- [format|2020 22 Player 1:%n(%u%n)*%nPlayer 2:%n(%u%n)*|]
let p1 = [Int] -> Seq Int
forall a. [a] -> Seq a
Seq.fromList [Int]
xs
let p2 = [Int] -> Seq Int
forall a. [a] -> Seq a
Seq.fromList [Int]
ys
print (score (play1 p1 p2))
print (score (snd (play2 Set.empty p1 p2)))
type Deck = Seq Int
score :: Deck -> Int
score :: Seq Int -> Int
score = [Int] -> Int
forall a. Num a => [a] -> a
forall (t :: * -> *) a. (Foldable t, Num a) => t a -> a
sum ([Int] -> Int) -> (Seq Int -> [Int]) -> Seq Int -> Int
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Int -> Int -> Int) -> [Int] -> [Int] -> [Int]
forall a b c. (a -> b -> c) -> [a] -> [b] -> [c]
zipWith Int -> Int -> Int
forall a. Num a => a -> a -> a
(*) [Int
1..] ([Int] -> [Int]) -> (Seq Int -> [Int]) -> Seq Int -> [Int]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. [Int] -> [Int]
forall a. [a] -> [a]
reverse ([Int] -> [Int]) -> (Seq Int -> [Int]) -> Seq Int -> [Int]
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Seq Int -> [Int]
forall a. Seq a -> [a]
forall (t :: * -> *) a. Foldable t => t a -> [a]
toList
type Rep = Vector Int
characterize :: Deck -> Deck -> Rep
characterize :: Seq Int -> Seq Int -> Rep
characterize Seq Int
xs Seq Int
ys = [Int] -> Rep
forall a. Unbox a => [a] -> Vector a
V.fromList (Seq Int -> [Int]
forall a. Seq a -> [a]
forall (t :: * -> *) a. Foldable t => t a -> [a]
toList Seq Int
xs [Int] -> [Int] -> [Int]
forall a. [a] -> [a] -> [a]
++ (-Int
1) Int -> [Int] -> [Int]
forall a. a -> [a] -> [a]
: Seq Int -> [Int]
forall a. Seq a -> [a]
forall (t :: * -> *) a. Foldable t => t a -> [a]
toList Seq Int
ys)
play1 :: Deck -> Deck -> Deck
play1 :: Seq Int -> Seq Int -> Seq Int
play1 Seq Int
Empty Seq Int
xs = Seq Int
xs
play1 Seq Int
xs Seq Int
Empty = Seq Int
xs
play1 (Int
x :<| Seq Int
xs) (Int
y :<| Seq Int
ys)
| Int
x Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
y = Seq Int -> Seq Int -> Seq Int
play1 (Seq Int
xs Seq Int -> Int -> Seq Int
forall a. Seq a -> a -> Seq a
:|> Int
x Seq Int -> Int -> Seq Int
forall a. Seq a -> a -> Seq a
:|> Int
y) Seq Int
ys
| Bool
otherwise = Seq Int -> Seq Int -> Seq Int
play1 Seq Int
xs (Seq Int
ys Seq Int -> Int -> Seq Int
forall a. Seq a -> a -> Seq a
:|> Int
y Seq Int -> Int -> Seq Int
forall a. Seq a -> a -> Seq a
:|> Int
x)
play2 :: Set Rep -> Deck -> Deck -> (Bool, Deck)
play2 :: Set Rep -> Seq Int -> Seq Int -> (Bool, Seq Int)
play2 Set Rep
_ Seq Int
Empty Seq Int
xs = (Bool
False, Seq Int
xs)
play2 Set Rep
_ Seq Int
xs Seq Int
Empty = (Bool
True, Seq Int
xs)
play2 Set Rep
seen xxs :: Seq Int
xxs@(Int
x :<| Seq Int
xs) yys :: Seq Int
yys@(Int
y :<| Seq Int
ys)
| Rep -> Set Rep -> Bool
forall a. Ord a => a -> Set a -> Bool
Set.member Rep
here Set Rep
seen = (Bool
True, Seq Int
xxs)
| Int
x Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Seq Int -> Int
forall a. Seq a -> Int
Seq.length Seq Int
xs, Int
y Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Seq Int -> Int
forall a. Seq a -> Int
Seq.length Seq Int
ys
, let x' :: Seq Int
x' = Int -> Seq Int -> Seq Int
forall a. Int -> Seq a -> Seq a
Seq.take Int
x Seq Int
xs
, let y' :: Seq Int
y' = Int -> Seq Int -> Seq Int
forall a. Int -> Seq a -> Seq a
Seq.take Int
y Seq Int
ys
, let x1 :: Int
x1 = Seq Int -> Int
forall a. Ord a => Seq a -> a
forall (t :: * -> *) a. (Foldable t, Ord a) => t a -> a
maximum Seq Int
x'
, let y1 :: Int
y1 = Seq Int -> Int
forall a. Ord a => Seq a -> a
forall (t :: * -> *) a. (Foldable t, Ord a) => t a -> a
maximum Seq Int
y'
= if Int
x1 Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
y1 Bool -> Bool -> Bool
|| (Bool, Seq Int) -> Bool
forall a b. (a, b) -> a
fst (Set Rep -> Seq Int -> Seq Int -> (Bool, Seq Int)
play2 Set Rep
forall a. Set a
Set.empty Seq Int
x' Seq Int
y')
then (Bool, Seq Int)
p1win
else (Bool, Seq Int)
p2win
| Int
x Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
y = (Bool, Seq Int)
p1win
| Bool
otherwise = (Bool, Seq Int)
p2win
where
here :: Rep
here = Seq Int -> Seq Int -> Rep
characterize Seq Int
xxs Seq Int
yys
seen1 :: Set Rep
seen1 = Rep -> Set Rep -> Set Rep
forall a. Ord a => a -> Set a -> Set a
Set.insert Rep
here Set Rep
seen
p1win :: (Bool, Seq Int)
p1win = Set Rep -> Seq Int -> Seq Int -> (Bool, Seq Int)
play2 Set Rep
seen1 (Seq Int
xs Seq Int -> Int -> Seq Int
forall a. Seq a -> a -> Seq a
:|> Int
x Seq Int -> Int -> Seq Int
forall a. Seq a -> a -> Seq a
:|> Int
y) Seq Int
ys
p2win :: (Bool, Seq Int)
p2win = Set Rep -> Seq Int -> Seq Int -> (Bool, Seq Int)
play2 Set Rep
seen1 Seq Int
xs (Seq Int
ys Seq Int -> Int -> Seq Int
forall a. Seq a -> a -> Seq a
:|> Int
y Seq Int -> Int -> Seq Int
forall a. Seq a -> a -> Seq a
:|> Int
x)