使用单例区分联合制作计算表达式
Making computation expressions using single case discriminated unions
如果我有一个尝试将起始数字相除两次的函数。整个工作流程必须 return 一个布尔值。
let divideBy bottom top =
if bottom = 0 then None
else Some (top/bottom)
let divideByWorkflow init x y =
match init |> divideBy x with
| None -> false
| Some a ->
match a |> divideBy y with
| None -> false
| Some b -> true
let good = divideByWorkflow 12 3 2
let bad = divideByWorkflow 12 3 0
下面的构建器是否正确?
type BoolMaybe = BoolMaybe with
member __.Bind (expr, f) =
match expr with
| Some value -> f value
| None -> false
member __.Return expr = expr
let divideByWorkflow init x y =
BoolMaybe {
let! a = init |> divideBy x
let! b = a |> divideBy y
return true
}
恕我直言,没有理由使用计算表达式使其变得更复杂。我宁愿使用一些简单的功能,例如:
let divideBy bottom top =
if bottom = 0 then None
else Some (top/bottom)
let divideByTwoTimes init x y =
init |> divideBy x |> Option.bind (divideBy y)
let divideByWorkflow init x = divideByTwoTimes init x >> Option.isSome
它有效,但我认为它不必要地具体。如果你确实想使用 CE,我宁愿定义一个普通的 Maybe
CE 然后做
let divideByWorkflow init x y =
Maybe {
let! a = init |> divideBy x
let! b = a |> divideBy y
return true
}
|> Option.defaultValue false
我同意 Dzoukr 的回答,我的看起来略有不同:
let divideBy bot top =
match bot with
| 0 -> None
| _ -> Some (top / bot)
let divideByWorkflow init x y =
Some init
|> Option.bind (divBy x)
|> Option.bind (divBy y)
|> Option.isSome
不需要计算表达式,在这里似乎没有任何好处。
如果我有一个尝试将起始数字相除两次的函数。整个工作流程必须 return 一个布尔值。
let divideBy bottom top =
if bottom = 0 then None
else Some (top/bottom)
let divideByWorkflow init x y =
match init |> divideBy x with
| None -> false
| Some a ->
match a |> divideBy y with
| None -> false
| Some b -> true
let good = divideByWorkflow 12 3 2
let bad = divideByWorkflow 12 3 0
下面的构建器是否正确?
type BoolMaybe = BoolMaybe with
member __.Bind (expr, f) =
match expr with
| Some value -> f value
| None -> false
member __.Return expr = expr
let divideByWorkflow init x y =
BoolMaybe {
let! a = init |> divideBy x
let! b = a |> divideBy y
return true
}
恕我直言,没有理由使用计算表达式使其变得更复杂。我宁愿使用一些简单的功能,例如:
let divideBy bottom top =
if bottom = 0 then None
else Some (top/bottom)
let divideByTwoTimes init x y =
init |> divideBy x |> Option.bind (divideBy y)
let divideByWorkflow init x = divideByTwoTimes init x >> Option.isSome
它有效,但我认为它不必要地具体。如果你确实想使用 CE,我宁愿定义一个普通的 Maybe
CE 然后做
let divideByWorkflow init x y =
Maybe {
let! a = init |> divideBy x
let! b = a |> divideBy y
return true
}
|> Option.defaultValue false
我同意 Dzoukr 的回答,我的看起来略有不同:
let divideBy bot top =
match bot with
| 0 -> None
| _ -> Some (top / bot)
let divideByWorkflow init x y =
Some init
|> Option.bind (divBy x)
|> Option.bind (divBy y)
|> Option.isSome
不需要计算表达式,在这里似乎没有任何好处。