Dev*_*r11 5 f# mailboxprocessor
是否可以等待邮箱处理器,以下代码在 F# 交互式中工作,但有没有办法在应用程序或单元测试中等待它?
[<TestMethod>]
member this.TestMailboxProcessor() =
let mailboxProcessor = MailboxProcessor<string>.Start(fun inbox ->
async {
while true do
let! msg = inbox.Receive()
printfn "agent got message %s" msg // too late, UnitTest exits
}
)
mailboxProcessor.Post "ping"
Console.WriteLine "message posted" // I see this in the console
Assert.IsTrue(true)
Run Code Online (Sandbox Code Playgroud)
在这种情况下这是不可能的,但您可以定义消息类型以包含AsyncReplyChannel<'t>,然后它允许您使用MailboxProcessor.PostAndReply而不是 Post。这样,调用代码可以(同步或异步)等待响应值,或者至少等待处理完成的指示。
您修改后的源代码可能如下所示:
[<TestMethod>]
member this.TestMailboxProcessor() =
let mailboxProcessor =
MailboxProcessor<string * AsyncReplyChannel<unit>>.Start(fun inbox ->
async {
while true do
let! msg, replyChannel = inbox.Receive()
printfn "agent got message %s" msg
(*
Reply takes a value of the generic param of
AsyncReplyChannel<'t>, in this case just a unit
*)
replyChannel.Reply()
}
)
(*
You can't create an AsyncReplyChannel<'t> value, but this does it for you.
Also always, always use timeouts when awaiting message replies.
*)
mailboxProcessor.PostAndReply(
(fun replyChannel -> "ping", replyChannel),
timeout = 1000)
(* This gets printed only after the message has been posted and processed *)
Console.WriteLine "message posted"
Assert.IsTrue(true)
Run Code Online (Sandbox Code Playgroud)
不过,MailboxProcessors 是一个有点棘手的话题,因此请确保始终使用超时,否则如果代码中出现错误或异常终止消息循环,您的代码将永远挂起。在测试中表现不佳,在生产中更糟。
您应该使用PostAndAsyncReplyor PostAndReply(阻止版本)
let replyAgent = MailboxProcessor.Start(fun inbox ->
let rec loop() =
async {
let! (replyChannel: AsyncReplyChannel<_>), msg = inbox.Receive()
replyChannel.Reply (sprintf "replied for message: %A" msg)
return! loop()
}
loop() )
let reply = replyAgent.PostAndReply(fun replCh -> replCh, "Hi")
printfn "%s" reply //prints "replied for message: Hi"
Run Code Online (Sandbox Code Playgroud)