在标准库中,add
函数具有以下签名:
val add : elt -> t -> t
所以我可以使用管道运算符添加元素:
Set.empty |> add elt1 |> add elt2
但是,当我切换到Core时,我注意到add的签名已经变为:
val add : ('a, 'cmp) t -> 'a -> ('a, 'cmp) t
现在set成为第一个参数。旧的管道风格不再适用于它。
使用Core Set添加元素的惯用方法是什么?
答案 0 :(得分:3)
不要使用管道来混淆功能应用程序!
管道有充分的理由存在,即构建管道,但在您的示例中,您只是滥用和混淆代码。
如果您只想将两个元素添加到集合中,只需写入
即可Set.(add (add empty a) b)
它非常干净清晰。如果您要添加更多元素, 您可以使用折叠函数来提高可读性:
List.fold_left Set.add Set.empty [a; b; c; d; e; f]
我们可以推测Jane Street正确地改变了Set.add
的签名,并将其与List.fold_left
一起使用,这是尾递归的,而List.fold_right
应该与来自标准库不是尾递归的。
答案 1 :(得分:2)
Jane Street在此blog post中将此设计选项解释为t comes first
规则。基本上重点是没有好的选择。有时将t
放在第一位,有时最后是有用的。也许他们选择的主要好处是它是一个决定。正如他们在介绍中所述,有时决定的好处是避免浪费时间一直思考这个问题。
更直接地回答你的问题,我会说没有惯用的解决方案。写出任何看似直观的内容。在其他答案中已经给出了一些建议。不要滥用管道操作员。如果要添加的项目更多,请使用fold
。我会避免flip
;你混淆代码而不会使它更简洁(但我有时也会使用它)。
最后,请注意Core
使用标记参数的额外设计选择通常可以完全解决问题。例如,他们的List.fold
也首先获取t
,但其他参数被标记。因此,您可以将t
放在任何您喜欢的位置,使List.fold
可用于管道和无管道。所以有人可能会问为什么他们没有签署Set.add
t -> elt:elt -> t
。好吧,标签也是一个额外的开销;它们会让您输入更多字符。在任何地方使用它们都是极端的,并且他们认为没有标签这个功能更好。
答案 2 :(得分:1)
在玩了几分钟后,我能想到的最好的方法是使用flip
来反转参数顺序。
let flip f a b = f b a
然后你可以写:
Set.empty |> flip add elt1 |> flip add elt2
上次检查时,flip
在Core中可用Fn.flip
。
(一般来说,在所有情况下,参数顺序都不是最好的。你最终可能会决定核心顺序对某些事情来说真的很棒。)