我正在努力正确地设置我的OCaml环境,以使用ppx派生程序映射、折叠和iter,如此处所述:https://github.com/ocaml-ppx/ppx_deriving#plugins-iter-map-and-fold。
我的最小示例在这里(我使用的是Base,因为这是我在更广泛的项目中使用的一个库):
open Base;;
type data = Row of float array | Dim of data array
[@@deriving iter, map, fold, show];;
let t = Row [|2.;2.|];;
pp_data Caml.Format.std_formatter t;;
map_data (fun x -> x +. 1.) t;;
pp_data Caml.Format.std_formatter t;;下面的代码是用ocamlfind ocamlc -package base -package ppx_deriving.iter -package ppx_deriving.map -package ppx_deriving.fold -package ppx_deriving.show -linkpkg -g test.ml && ./a.out编译的;我得到一个编译错误,说明map_data有data -> data类型。但是根据文档和我的一般知识,map得到了一个函数和一个可映射的结构,而这里的情况似乎并非如此。在utop中测试这一点会给我同样的错误。
我遗漏了什么吗?
(预先谢谢:)
发布于 2020-04-08 13:13:17
这些派生器处理多态数据结构,并将用户函数应用于与该结构的类型变量相对应的所有值。由于您没有任何类型变量,所以生成的map_data函数有缺陷,但是非常自然,因为缺少类型变量意味着一个常量函数。
换句话说,对于一些带有map_x类型变量的多态type ('s1, ..., 'sN) x,N函数的一般结构是
('s1 -> 't1) -> ... -> ('sN -> 'tN) -> ('s1,...,'sN) x -> ('t1,...,'tN) x也就是说,对于每个类型变量,它期望一个函数将该类型的值映射到其他类型,这样在您的情况下,映射函数的参数数是N+1,因为您有零类型变量,所以没有映射函数,所以只有x -> x。
如果要将您的类型重新定义为
type 'a data = Row of 'a array | Dim of 'a data array
[@@deriving iter, map, fold, show]您将得到具有expect类型的map_data ('a -> 'b) -> 'a data -> 'b data。派生器甚至会理解数组是一种数据结构,并对其进行递归,例如,
let input = Dim [|Row [|1;2;3|]; Row [|3;4;5|]|]
map_data (fun x -> x + 1) input;;
- : int data = Dim [|Row [|2; 3; 4|]; Row [|4; 5; 6|]|]当然,如果您不希望在接口中使用多态类型,则始终可以创建类型别名,例如,
type t = float data并将map_data公开为
val map_data : (float -> float) -> t -> thttps://stackoverflow.com/questions/61100710
复制相似问题