OCaml:为什么`let f : int -> int list -> int list = (::);;` 会失败?

问题描述

我认为 OCaml 会阅读

let f : int -> int list -> int list = (::);;

作为部分应用的实例,并将函数存储为f。但是,编译器抱怨该函数应用于太少的参数。对此的任何见解将不胜感激,谢谢。

解决方法

OCaml 中的值 :: 是一个构造函数而不是一个通用函数。因此,它需要以下参数(在括号中)。您可以在 (::) 形式中使用它,但必须后跟括号中的两个值:

# ( :: ) (3,[]);;
- : int list = [3]

其他构造函数也是如此,比如 Some:

# let f : 'a -> 'a option = Some
Error: The constructor Some expects 1 argument(s),but is applied here to 0 argument(s)

你当然可以这样定义你的函数:

let f : int -> int list -> int list = fun a b -> (::) (a,b)

或者这个:

let f : int -> int list -> int list = fun a b -> a :: b

甚至像这样:

let f : int -> int list -> int list = List.cons

(其他语言,尤其是 Haskell,将构造函数视为函数。因此您可以使用部分应用的构造函数等。在我看来,这在美学上更优越,即看起来更优雅。)