为什么我的邮箱处理器挂起了?

Mar*_*son 3 f# mailboxprocessor

我无法弄清楚为什么下面的代码挂在调用GetTotal.我似乎无法在MailboxProcessor内部进行调试,因此很难看到发生了什么.

module Aggregator

open System

type Message<'T, 'TState> =
    | Aggregate of 'T
    | GetTotal of AsyncReplyChannel<'TState>

type Aggregator<'T, 'TState>(initialState, f) =
    let myAgent = new MailboxProcessor<Message<'T, 'TState>>(fun inbox ->
        let rec loop agg =
            async {
                let! message = inbox.Receive()
                match message with
                    | Aggregate x -> return! loop (f agg x)
                    | GetTotal replyChannel ->
                        replyChannel.Reply(agg)
                        return! loop agg
            }
        loop initialState
        )

    member m.Aggregate x = myAgent.Post(Aggregate(x))
    member m.GetTotal = myAgent.PostAndReply(fun replyChannel -> GetTotal(replyChannel))

let myAggregator = new Aggregator<int, int>(0, (+))

myAggregator.Aggregate(3)
myAggregator.Aggregate(4)
myAggregator.Aggregate(5)

let totalSoFar = myAggregator.GetTotal
printfn "%d" totalSoFar

Console.ReadLine() |> ignore
Run Code Online (Sandbox Code Playgroud)

当直接使用相同的MailboxProcessor而不是包装在Aggregator类中时,它似乎工作正常.

Tom*_*cek 7

问题是你没有启动代理.您可以Start在创建代理后调用:

let myAgent = (...)
do myAgent.Start()
Run Code Online (Sandbox Code Playgroud)

或者,您可以使用MailboxProcessor<'T>.Start而不是调用构造函数来创建代理(我通常更喜欢这个选项,因为它看起来更实用):

let myAgent = MailboxProcessor<Message<'T, 'TState>>.Start(fun inbox ->  (...) )
Run Code Online (Sandbox Code Playgroud)

我想您无法调试代理,因为代理内部的代码实际上并未运行.我尝试printfn "Msg: %A" message在调用Receive代理内部之后立即添加(打印传入的消息以进行调试)并且我注意到,在调用之后Aggregate,代理实际上没有收到任何消息...(它仅在调用后被阻止GetTotal,这样才能获得回复)

作为旁注,我可能会GetTotal变成一种方法,所以你打电话GetTotal().每次访问属性时都会重新评估属性,因此您的代码执行相同的操作,但最佳做法不建议使用执行复杂工作的属性.