我编写了一个 FsCheck 生成器,它产生随机 glob 语法模式(例如a*c?
)以及与该模式匹配的随机字符串(例如abcd
)。但是我的解决方案使用了一个可变变量,我对此感到很惭愧。看一看:
open FsCheck
type TestData = {Pattern: string; Text: string}
let stringFrom alphabet =
alphabet |> Gen.elements |> Gen.listOf |> Gen.map (List.map string >> List.fold (+) "")
let singleCharStringFrom alphabet =
alphabet |> Gen.elements |> Gen.map string
let matchingTextAndPatternCombo = gen {
let toGen = function
| '*' -> stringFrom ['a'..'f']
| '?' -> singleCharStringFrom ['a'..'f']
| c -> c |> string |> Gen.constant
let! pattern = stringFrom (['a'..'c']@['?'; '*'])
let mutable text = ""
for gen in Seq.map toGen pattern do
let! textPart = gen
text <- text + textPart
return {Pattern = pattern; Text = text}
}
请注意,它text
是可变的,以及它的值是如何在循环中累积的。
我的直觉告诉我,必须有一种方法可以fold
将生成器导入text
,但我不知道如何,因为我不了解let!
引擎盖下的工作原理(还)。我正在考虑类似于以下内容:
let! text = pattern |> Seq.map toGen |> Seq.fold (?) (Gen.constant "")
我在正确的轨道上吗?累加器和种子应该是什么fold
样的?