本文是一个系列,是函数式Swift的读书笔记(其实是为了备忘)
测试通常由一些代码片段和预期结果组成。执行代码之后,将它的结果与测试中定义的预期结果相比较。
QuickCheck (Claessen and Hughes 2000) 是一个用于随机测试的 Haskell 库,我们用Swift部分地 移植它。
// 之前对测试一直没了解过。刚好看一下。
1. 构建QuickCheck
需要以下四步:
- 需要一个生成不同类型随机数的方法
- 实现check函数,传入随机数。
- 测试失败了,我们希望测试的输入值尽可能小。
- 做一些额外的工作确保 检验函数适用于带有泛型的类型
1.1 随机数生成
//定义一个生产随机数的协议
protocol Arbitrary {
static func arbitrary()->Self
}
//适用
//Int.arbitrary()
// 为避免随机整数越界,添加一个变量进行约束:
extension Int{
static func arbitrary(in range:CountableRange<Int>)->Int{
let diff = range.upperBound - range.lowerBound
return range.lowerBound + (Int.arbitrary()%diff)
}
}
// 生成随机字符串
extension UnicodeScalar:Arbitrary{
static func arbitrary() -> UnicodeScalar {
return UnicodeScalar(Int.arbitrary(in: 65..<90))!
}
}
//生成随机字符串
extension String:Arbitrary{
static func arbitrary() -> String {
let randomLength = Int.arbitrary(in: 0..<40)
let randomScalars = (0..<randomLength).map{
_ in
UnicodeScalar.arbitrary()
}
return String(UnicodeScalarView(randomScalars))
}
}
String.arbitrary() // 直接生成了0~40位之间的,随机大写字符串。
1.2 实现check函数
check检验函数的第一个版本,check1: 包含一个简单循环,每次随机输入值,发现问题打印,否知汇报成功通过的次数。
func check1<A:Arbitrary>(_ message:String,_ property:(A)->Bool) ->(){
for _ in 0 ..< numberOfIterations {
let value = A.arbitrary()
guard property(value) else{
print("\"\(message)\" doesn't hold:\(value)")
return
}
}
print("\"\(message) passed \(numberOfIterations) test")
}
extension CGSize{
var area : CGFloat{
return width * height
}
}
extension CGSize:Arbitrary{
static func arbitrary() -> CGSize {
return CGSize(width: .arbitrary(), height: .arbitrary())
}
}
//测试
check1("Area should be at least 0", {
(size: CGSize) in
size.area > 0
})
//当随机到有负的值的时候,测试就通不过。
1.3 缩小范围
// 如果在字符串上运行check1, 可能会受到很长的失败消息:
check1("every string starts with hello"){
(s:String) in
s.hasPrefix("hello")
}
通常,反例所处的范围越小,越容易定位到失败是由哪一段代码引起的