虽然如果您提供 的实现会有所帮助gen
,但我猜它是这样的:
gen :: a -> ([a] -> [a]) -> ([a] -> Bool) -> a
gen init next stop = loop [init]
where
loop xs | stop xs = head xs
| otherwise = loop (next xs)
您要测试的属性next
是永远不会提供空列表的属性。测试这个的一个障碍是你想检查一个内部循环不变量 inside gen
,所以这需要从外部可用。让我们修改gen
以返回此信息:
genWitness :: a -> ([a] -> [a]) -> ([a] -> Bool) -> (a,[[a]])
genWitness init next stop = loop [init]
where
loop xs | stop xs = (head xs,[xs])
| otherwise = second (xs:) (loop (next xs))
我们second
使用
Control.Arrow。原件gen
很容易定义为genWitness:
gen' :: a -> ([a] -> [a]) -> ([a] -> Bool) -> a
gen' init next stop = fst (genWitness init next stop)
由于惰性评估,这不会给我们带来太多开销。回物业!为了能够显示 QuickCheck 生成的函数,我们使用模块
Test.QuickCheck.Function。虽然这里不是绝对必要的,但一个好习惯是单态属性:我们使用Int
s 的列表,而不是允许单态限制使它们成为单元列表。现在让我们声明一下属性:
prop_gen :: Int -> (Fun [Int] [Int]) -> (Fun [Int] Bool) -> Bool
prop_gen init (Fun _ next) (Fun _ stop) =
let trace = snd (genWitness init next stop)
in all (not . null) trace
让我们尝试使用 QuickCheck 运行它:
ghci> quickCheck prop_gen
有些东西似乎在循环......当然是的:gen
如果stop
在列表中从next
永远不会循环True
!让我们改为尝试查看输入跟踪的有限前缀:
prop_gen_prefix :: Int -> (Fun [Int] [Int]) -> (Fun [Int] Bool) -> Int -> Bool
prop_gen_prefix init (Fun _ next) (Fun _ stop) prefix_length =
let trace = snd (genWitness init next stop)
in all (not . null) (take prefix_length trace)
我们现在很快得到一个反例:
385
{_->[]}
{_->False}
2
第二个函数是参数next
,如果它返回空列表,那么循环gen
将给出next
一个空列表。
我希望这能回答这个问题,并让您了解如何使用 QuickCheck 测试高阶函数。