如何解决在 OCaml 中做符号
OCaml 是否与 Haskell 的 Do Notation 等效?
另一种说法 - 有没有一种简单的方法可以更轻松地处理嵌套的 monadic 操作...因为这很烦人:
open Batteries
open BatResult.Infix
let () =
let out =
(Ok("one")) >>=
(fun a ->
let b = Ok("two") in
b >>= (fun c ->
print_endline(a);
print_endline(c);
Ok(c)
)) in
ignore(out)
解决方法
是的,从 OCaml 4.08 开始,可以描述 let
运算符。例如:
let (let*) x f = BatResult.bind x f
let (let+) x f = BatResult.map f x
这使得以接近直接风格的风格编写程序成为可能:
let out =
let* a = Ok "one" in
let* b = Ok "two" in
let () = print_endline a in
let () = print_endline b in
Ok b
您可以描述大量的运算符,用于 monads(例如 let*
的 bind
和 let+
的 map
),用于应用的(例如 let+
为 map
和 and+
为 product
(或 zip
))等。
否则,可以使用语法扩展,例如 https://github.com/janestreet/ppx_let,目前,它提供了比 let 运算符更多的可能性。如果您想要 let 运算符的示例,在 Preface(无耻插件)中,我们定义了很多!
edit:正如@ivg 所说,您可以使用任何 $ ∣ & ∣ * ∣ + ∣ - ∣ / ∣ = ∣ > ∣ @ ∣ ^ ∣ |
来定义您的 let
或 and
运算符。
见:https://ocaml.org/manual/bindingops.html
,除非您特别尝试混淆您的代码,否则编写没有 do-notation 的 monadic 代码非常容易,例如,如果我们从您的示例中删除过多的1 括号,它已经非常易读了,
let v0 =
Ok "one" >>= fun a ->
Ok "two" >>= fun c ->
print_endline a;
print_endline c;
Ok c
如果我们使用 let-binding 操作符,例如使用 monads [1,2] 库(也是一个无耻的插件),那么我们可以写得更简洁,>
open Monads.Std
open Monad.Result.Error.Syntax
open Monad.Result.Error.Let
let v1 =
let+ a = Ok "one" and+ b = Ok "two" in
print_endline a;
print_endline b;
b
1) OCaml 通过并列表示应用程序操作(函数对其参数的应用程序和应用程序对其参数的构造函数),例如,sin x
(不是 {{ 1}}) 或 sin(x)
(不是 Ok 42
),而且应用运算符比中缀运算符具有更高的优先级(绑定更紧密),因此您可以编写 Ok (42)
。 lambda 运算符 sin x > cos x
与一元运算符很好地链接在一起,例如,
fun <var> -> <expr>
与
相同x >>= (fun x -> y >>= (fun y -> x + y))
或者,更常见的写成
x >>= fun x -> y >>= fun y -> x + y
作为对 lambda(抽象)运算符的 let-legacy 的致敬,参见,
x >>= fun x ->
y >>= fun y ->
x + y
和最新版本的 OCaml 甚至允许您在右侧使用双关语,例如,
let* x = x in
let* y = y in
x + y
甚至
let* x in
let* y in
x + y
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。