12

我有一个类型类Atomic,它定义了用于将某些类型转换为/从包装器值 ( Atom) 转换的函数。我想定义一个 QuickCheck 属性,它声明:“对于 的所有实例Atomic,任何值都可以安全地存储和检索”。该属性如下所示:

class Atomic a where
    toAtom :: a -> Atom
    fromAtom :: Atom -> Maybe a

prop_AtomIdentity x = fromAtom (toAtom x) == Just x

但是,如果我只是尝试通过 QuickCheck 运行该属性,它只会选择一个实例 ( Bool) 并对其进行测试。我目前正在通过为测试列表中每个支持的原子类型定义类型签名来解决这个问题,但这很冗长且容易出错:

containerTests =
    [ run (prop_AtomIdentity :: Bool -> Bool)
    , run (prop_AtomIdentity :: Word8 -> Bool)
    , run (prop_AtomIdentity :: String -> Bool)
    {- etc -} ]

我正在尝试定义一个自动执行此操作的函数:

forallAtoms :: (Atomic a, Show a) => (a -> Bool) -> [TestOptions -> IO TestResult]
forallAtoms x =
    [ run (x :: Bool -> Bool)
    , run (x :: Word8 -> Bool)
    , run (x :: String -> Bool)
    {- etc -} ]

containerTests = forallAtoms prop_AtomIdentity

但它失败并出现类型检查错误:

Tests/Containers.hs:33:0:
    Couldn't match expected type `Word8' against inferred type `String'
    In the first argument of `run', namely `(x :: Word8 -> Bool)'
    In the expression: run (x :: Word8 -> Bool)
    In the expression:
        [run (x :: Bool -> Bool), run (x :: Word8 -> Bool),
         run (x :: String -> Bool)]

有没有更好的方法来针对多种类型测试 QC 属性?如果没有,可以使 forallAtoms 工作还是类型系统不支持?

4

1 回答 1

12

我无法编译你的代码,所以......盲拍:

尝试

forallAtoms :: (forall a. (Atomic a, Show a) => a -> Bool) -> [TestOptions -> IO TestResult]

作为类型签名。这需要 -XRankNTypes 语言扩展。

正如我所看到的,您遇到的问题是 GHC 试图为整个函数范围寻找一种类型来插入,但您已经在那里给出了三种不同的类型ax :: (a -> Bool)

于 2009-09-19T18:40:33.913 回答