如果没有DataKinds,如何使值依赖于其他值?

时间:2015-08-20 14:58:35

标签: haskell data-kinds

我有一个Universe类型和一个worker类型。工人可以改变宇宙。我想要实现的是确保宇宙只能由来自该宇宙的工人(不是将来或​​过去的工人)修改。

我能做到的最好的就是:

{-# LANGUAGE DataKinds       #-}
{-# LANGUAGE KindSignatures  #-}

module Module(initialUniverse, doSomething, doStep) where

data Nat = Z | S Nat

data Universe (t :: Nat) = Universe {
  _allWorkers :: [Worker t]
}

data Worker (t :: Nat) = Worker

initialUniverse :: Universe Z
initialUniverse = Universe [Worker]

doSomething :: Worker t -> Universe t -> Universe (S t)
doSomething = undefined

doStep :: Universe t -> Universe (S (S t))
doStep u = let w = head $ _allWorkers u
               u2 = doSomething w u
               w2 = head $ _allWorkers u2
               u3 = doSomething w2 u2
           in u3

如果我将w2 = head $ _allWorkers u2更改为w2 = head $ _allWorkers u,我会收到编辑错误,这就是我想要的。

我不喜欢的是现在我有一个版本附加到每个宇宙,我必须手动增加它。这可以用另一种不需要显式版本控制的方式完成吗?使doSomething函数返回Universe otherType类似检查器知道otherType的内容与t不同。

感谢您的时间。

1 个答案:

答案 0 :(得分:5)

一种方法是使用存在类型:

data Un = forall s. Un (Universe s)

doSomething :: Worker s -> Universe s -> Un

initialUniverse :: Un


doStep :: Un -> Un
doStep (Un u) = let w = head $ _allWorkers u
               u2 = doSomething w u
               w2 = head $ _allWorkers u2
               u3 = doSomething w2 u2
            in Un u3