gpt4 book ai didi

haskell - 异构 GADT 列表

转载 作者:行者123 更新时间:2023-12-02 10:40:32 26 4
gpt4 key购买 nike

我有一个如下所示的 GADT

data MyTypes
= MyInt
| MyDouble

data Test (t :: MyTypes) where
A :: Int -> Test 'MyInt
B :: Double -> Test 'MyDouble

这使我能够在类型级别跟踪 Test 值中包含的值,以便我也可以执行类似的操作

data Test2 (t :: MyTypes) where
A2 :: Test 'MyInt -> Test2 'MyInt
B2 :: Test 'MyDouble -> Test2 'MyDouble

并传递信息。

但是,如果我想要具有不同 MyTypesTest 值列表,例如

myData :: [Test (t :: MyTypes)]
myData =
[ A (3 :: Int)
, B (5.0 :: Double)
]

我得到了预期的‘t’是一个刚性类型变量绑定(bind)错误消息。

我尝试使用存在类型来克服严格类型变量的问题,但随后我失去了传递有关 MyType 的类型级别信息的能力。

遇到这样的问题我该如何解决?

最佳答案

这里的解决方案是存在主义的:

data Test a where
A :: Int -> Test Int
B :: Double -> Test Double

data SomeTest where
SomeTest :: Test a -> SomeTest

myData :: [SomeTest]
myData =
[ SomeTest (A (3 :: Int))
, SomeTest (B (5.0 :: Double))
]

这只会改变您使用这种类型的方式。您可以通过模式匹配来恢复类型信息:

consume :: Test a -> Int
consume (A a) = a + 1
consume (B b) = truncate b

map (\ (SomeTest x) -> consume x) myData :: [Int]

使用 RankNType,您可以使用恢复类型的延续来解开它:

test :: (forall a. Test a -> r) -> SomeTest -> r
test k (SomeTest x) = k x

test (\ x -> case x of
A a -> a + 1 {- ‘a’ is known to be ‘Int’ here -}
B b -> truncate b {- ‘b’ is known to be ‘Double’ here -})
:: SomeTest -> Int

当您使用它来将多个事物打包在某种“模块”中时,没有任何类型类约束的存在性非常有用,其中它们都必须就类型达成一致,但该类型是从外面看不透明。这限制了消费者可以执行的操作 - 例如,考虑一对请求和一个用于存储该请求结果的变量:

data SomeRequest where
SomeRequest :: IO a -> IORef a -> SomeRequest

fetchRequests :: [SomeRequest] -> IO ()
fetchRequests = traverse_ fetchRequest
where

-- ‘fetchRequest’ controls the fetching strategy (sync/async)
-- but can’t do anything with the fetched value
-- other than store it in the ‘IORef’.
fetchRequest :: SomeRequest -> IO ()
fetchRequest (SomeRequest request result) = do
value <- request
writeIORef result value

如果你有一个完全多态的类型,比如:

data Test a where
Test :: a -> Test a

然后您可以通过添加类型类约束来恢复有关类型的更多有趣信息。例如,如果您想要完整的动态信息,可以使用 Typeable 获取:

data SomeTest where
SomeTest :: Typeable a => Test a -> SomeTest

test :: (forall a. Typeable a => Test a -> r) -> SomeTest -> r
test k (SomeTest x) = k x

test (\ (Test a) -> case cast a of
Just a' -> (a' :: Int) + 1
Nothing -> case cast a of
Just a' -> length (a' :: String)
Nothing -> 0)

大多数时候,您可以使用比这更弱的类型类,具体取决于您实际需要的操作。

关于haskell - 异构 GADT 列表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49519247/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com