我认为是时候尝试 FsCheck 了,但事实证明它比我想象的要难。有很多关于Arb
, generators 等的文档,但似乎没有任何关于如何应用这些知识的指导。或者我只是不明白。
可能更难理解的是,测试、属性、生成器、任意性、收缩以及在我的情况下随机性(一些测试自动生成随机数据,其他测试不会)之间的关系对我来说并不清楚。我没有 Haskell 背景,所以这也无济于事。
现在的问题是:如何生成随机整数?
我的测试场景可以用乘法的性质来解释,比如说分布性:
static member ``Multiplication is distributive`` (x: int64) y z =
let res1 = x * (y + z)
let res2 = x * y + x * z
res1 = res2
// run it:
[<Test>]
static member FsCheckAsUnitTest() =
Check.One({ Config.VerboseThrowOnFailure with MaxTest = 1000 }, ``Multiplication is distributive``)
当我使用Check.Verbose
NUnit 集成运行它时,我得到如下测试序列:
0:
(-1L, -1L, -1L)
1:
(-1L, -1L, 0L)
2:
(-1L, -1L, -1L)
3:
(-1L, -1L, -1L)
4:
(-1L, 0L, -1L)
5:
(1L, 0L, 2L)
6:
(-2L, 0L, -1L)
7:
(-2L, -1L, -1L)
8:
(1L, 1L, -2L)
9:
(-2L, 2L, -2L)
经过1000次测试,它还没有结束100L
。不知何故,我想象这将“自动”选择均匀分布在整个范围内的随机数int64
,至少我是这样解释文档的。
既然没有,我开始尝试并想出如下愚蠢的解决方案来获得更高的数字:
type Generators =
static member arbMyRecord =
Arb.generate<int64>
|> Gen.where ((<) 1000L)
|> Gen.three
|> Arb.fromGen
但这变得异常缓慢,显然不是正确的方法。我确信必须有一个我缺少的简单解决方案。我试过了Gen.choose(Int64.MinValue, Int64.MaxValue)
,但这只支持整数,而不是长整数(但即使只有整数我也无法让它工作)。
最后,我需要一个适用于所有原始数值数据类型的解决方案,包括它们的最大值和最小值、它们的零和一,以及从其中的任何内容中进行的一些随机选择。