现在我有以下模式匹配
let argList = args |> List.ofSeq
match argList with
| "aaa" :: [] -> run "aaa"
| "bbb" :: DateTimeExact "yyyyMMdd" date :: [] -> run "bbb" date
....
它用于解析命令行参数,例如exec aaa
exec bbb 20141101
现在我希望能够添加选项
-o
(选修的)。如exec bbb 20141101 -o
.如何修改模式来做到这一点?更好的是,-o
应该可以放在任何位置。
最佳答案
如果您正在寻找一种可以轻松扩展的方法,我发现在解释命令行参数时使用 F# 类型系统真的很有帮助。
最终,您希望获得 argList 中提供的详细信息。所以首先,定义你想要的类型。然后,您将字符串列表解析为这些类型。在您的示例中,您似乎可以使用“aaa”或“bbb”,然后紧跟日期。感觉就像是 Union 类型:
type Cmd =
| Aaa
| Bbb of DateTime
此外,可能有一个通过“-o”的额外标志可以出现在任何时候(除了 bbb 和它的日期之间)。所以感觉就像一个记录类型:
type Args = { Cmd: Cmd; OSet: bool; }
所以现在我们知道我们想要将 argList 集合变成 Args 的一个实例。
您想要遍历 argList 中的每个项目并处理它们。一种方法是使用递归。在继续检查列表的其余部分之前,您希望匹配每个项目(或一对项目,或三倍等)。在每场比赛中,您都会更新* Args 以包含新的详细信息。
(* 因为 Args 是不可变的,所以您实际上并没有更新它。创建了一个新实例。)
let rec parseArgs' argList (a: Args) =
match argList with
| [] ->
a
| "aaa"::xs ->
parseArgs' xs { a with Cmd = Aaa }
| "bbb":: DateTimeExact "yyyyMMdd" d ::xs ->
parseArgs' xs { a with Cmd = Bbb(d) }
| "-o"::xs ->
parseArgs' xs { a with OSet = true }
| x::_ ->
failwith "Invalid argument %s" x
调用 parseArgs' 时,您需要提供 Args 的初始版本 - 这是您的“默认”值。
let parseArgs argList = parseArgs' argList { Cmd = Aaa; OSet = false }
然后你可以使用它:
parseArgs ["aaa"] // valid
parseArgs ["bbb"; "20141127"] // valid
parseArgs ["bbb"; "20141127";"-o"] // valid
parseArgs ["-o";"bbb"; "20141127"] // valid
parseArgs ["ccc"] // exception
您的 Args 实例现在具有强类型形式的所有详细信息。
let { Cmd = c; OSet = o } = parseArgs ["aaa"]
match c with
| Aaa -> run "aaa" o
| Bbb d -> run "bbb" d o
当您需要更多不同的选项时,您只需将它们添加到您的类型中,然后更新您的匹配语句以处理任何输入版本。
当然有很多不同的方法来处理这个问题。您可能是错误消息而不是异常。您可能希望 Cmd 值是一个选项,而不是默认为“Aaa”。等等等等。
编辑回应评论:
添加一个额外的
-p someValue
很简单。一、更新
Args
来保存新数据。在您的示例中,“someValue”的值是重要的部分,但可能会或可能不会提供。它由“-p”定义,以便我们看到它时就知道它。为简单起见,我将假设 someValue 是 string
.所以Args
变成:type Args = { Cmd: Cmd; OSet: bool; PValue: string option }
在
Args
中添加新字段后,您的“默认”应该提示,因为未设置新字段。我要说的是,默认情况下, -p 没有设置(这就是我使用 string option
的原因)。所以更新为:let parseArgs argList = parseArgs' argList { Cmd = Aaa; OSet = false; PValue = None }
然后,您只需要在提供值(value)时识别并捕获值(value)。这是模式匹配部分:
let rec parseArgs' argList (a: Args) =
match argList with
... snip ...
| "-p"::p::xs ->
parseArgs' xs { a with PValue = Some p}
... snip ...
然后,当您拥有
Args
时,只需使用 PValue 值即可。元素。注:我在这里没有做太多验证 - 我只是假设“-p”之后的任何内容都是我想要的值。您可以在模式匹配期间使用“何时”保护添加验证,或验证
Args
创造后的值(value)。你走多远取决于你。我通常会考虑我的听众(只是我、同事、整个互联网)。允许单独指定“bbb”和日期是否是一个好主意取决于您作为设计者,但如果日期应该只用“bbb”命令指定并且是强制性的,那么它可能会使感觉让他们在一起。
如果 Cmd 和日期没有“紧密联系”,您可以将日期与 cmd 分开进行模式匹配。如果是这样,您可能会将日期从保留在
Cmd
上移开。联合,成为 Args
中的一个领域.如果日期是可选的,您可以使用
-d [date]
选项。这将保持与其他可选参数相同的模式。一个重要的目标是尝试使您的界面尽可能直观以供人们使用。这主要是关于“可预测”。这并不一定意味着要满足尽可能多的输入样式。
关于F# 可选模式匹配,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27162077/