Пытаясь понять экзистенциальные типы в Haskell, я решил реализовать векторный тип данных фиксированной длины на основе целых чисел. Я использую ghc 7.8.3.
В частности, я хотел написать программу, которая запрашивает у пользователя возможное новое значение для добавления к вектору фиксированной длины, а затем отображает результирующий вектор.
Сначала я написал первую версию программы следующим образом:
{-# LANGUAGE GADTs #-}
{-# LANGUAGE ExistentialQuantification #-}
{-# LANGUAGE Rank2Types #-}
import System.IO (hFlush, stdout)
data Z
data S n
data Vect n where
ZV :: Vect Z
CV :: Int -> Vect n -> Vect (S n)
data AnyVect = forall n. AnyVect (Vect n)
instance Show (Vect n) where
show ZV = "Nil"
show (CV x v) = show x ++ " : " ++ show v
vecAppend :: Int -> Vect n -> Vect (S n)
vecAppend x ZV = CV x ZV
vecAppend x v = CV x v
appendElem :: AnyVect -> IO AnyVect
appendElem (AnyVect v) = do
putStr "> "
hFlush stdout
x <- readLn
return $ if x == 0 then AnyVect v else AnyVect $ vecAppend x v
main = do
AnyVect v <- appendElem $ AnyVect ZV
putStrLn $ show v
который работает, как ожидалось. Тогда я решил избавиться от ненужного AnyVect:
{-# LANGUAGE GADTs #-}
{-# LANGUAGE ExistentialQuantification #-}
{-# LANGUAGE Rank2Types #-}
import System.IO (hFlush, stdout)
data Z
data S n
data Vect n where
ZV :: Vect Z
CV :: Int -> Vect n -> Vect (S n)
instance Show (Vect n) where
show ZV = "Nil"
show (CV x v) = show x ++ " : " ++ show v
vecAppend :: Int -> Vect n -> Vect (S n)
vecAppend x ZV = CV x ZV
vecAppend x v = CV x v
appendElem :: Vect n -> (forall n. Vect n -> a) -> IO a
appendElem v f = do
putStr "> "
hFlush stdout
x <- readLn
return $ if x == 0 then f v else f $ vecAppend x v
main = do
appendElem ZV show >>= putStrLn
который тоже работает, даже мне не очень нравится, как написан main.
Есть ли другой более простой/чистый способ написать это?
vecAppend = CV
более прямой. Я бы также переименовал его вvecCons
, так как добавление звучит так, как будто оно работает с двумя векторами. - person chi   schedule 08.09.2014