Полиморфный вариант ограничения в определении варианта
Предположим, что я хочу смоделировать простой тип выражения в OCaml:
type expr =
| `Int of int
| `Str of string
| `IntAdd of expr * expr
| `StrAdd of expr * expr
Можно ли ограничить expr
в expr * expr
для конкретных конструкторов expr
сам (то есть я хотел бы IntExpr
разрешить только Int)? Я могу имитировать это с сопоставлением с образцом, но это становится громоздким после expr
расширяется. Можно ли как-то использовать систему типов OCaml для достижения этой цели?
Я попытался использовать верхние границы полиморфного типа следующим образом:
type expr =
| `Int of int
| `Str of string
| `IntAdd of [< `Int] * [< `Int]
| `StrAdd of [< `Str] * [< `Str]
но компилятор не принимает это (с сообщением In case IntAdd of [< Int ] * ([< Int ] as 'a) the variable 'a is unbound
). Есть ли хитрость, чтобы заставить это работать?
1 ответ
Данный пример достаточно прост для полиморфных вариантов:
type int_expr = [`Int of int | `Add of int_expr * int_expr]
type string_expr = [`String of string | `Concat of string_expr * string_expr]
type expr = [int_expr | string_expr]
Если вам нужны более интересные функции, такие как полиморфные структуры данных, необходимы GADT:
type _ expr =
| Int : int -> int expr
| Add : int expr * int expr -> int expr
| String : string -> string expr
| Concat : string expr * string expr -> string expr
| Pair : 'a expr * 'b expr -> ('a * 'b) expr
| Fst : ('a * 'b) expr -> 'a expr
| Snd : ('a * 'b) expr -> 'b expr
Вывод и понятность сообщений об ошибках страдают с GADT, поэтому будьте готовы работать, чтобы преодолеть эти трудности, если вы решите их использовать.