程序可能由于lambda闭包或调度程序问题而挂起

时间:2017-12-21 01:50:56

标签: c# callback closures dispatcher ui-thread

我实现了一个实验性的轻量级内存消息总线,收件人可以通过下面粘贴的Subscribe()方法订阅消息。

发件人可以发送消息,消息总线将调用内部回拨internalCallback。在调用回调之前,可以将消息进行深度克隆并封送到UI线程上。

这是我的问题出现的地方:当我注释掉UI调度程序时(如下面的代码段中所做),然后正确调用回调。在调度程序处于活动状态时,整个应用程序将挂起(不会引发运行时错误)。让我更加困惑的是,当我在UI线程上运行调用方法时,整个下面的代码工作得非常好。但是现在整个框架也可以在不同的任务/线程上发送消息,这就是出现问题的时候。

我应该研究或调整什么?

由于

public void Subscribe<T>(string subscriberId, string topic, Action<string, T> callback, bool returnOnUiThread, bool makeDeepCopy, bool catchAll)
    {
        //create new peer connection if it does not yet exist
        if (!_peerConnections.ContainsKey(subscriberId))
        {
            var newPeer = new PeerConnection(subscriberId)
            {
                ConnectionStatus = PeerConnectionStatus.Connected,
                CreationTimeStamp = DateTime.Now,
                LastAliveTimeStamp = DateTime.Now
            };

            _peerConnections.Add(subscriberId, newPeer);
        }

        var internalCallBack = new Action<string, object>((header, msg) =>
        {
            //make a deep copy via serialization if requested
            if (makeDeepCopy == true)
            {
                //try deep clone
                var serializedObject = Serializers.JsonStringFromObject(msg);
                msg = Serializers.ObjectFromJsonString<T>(serializedObject);
            }

            var handle = callback;
            handle(header, (T)msg);

            ////return on ui thread if requested
            //if (returnOnUiThread == true)
            //{
            //    Application.Current.Dispatcher.Invoke(() =>
            //    {
            //        var handle = callback;
            //        handle(header, (T)msg);
            //    });
            //}
            //else
            //{
            //    var handle = callback;
            //    handle(header, (T)msg);
            //}
        });

        //adding subscription to collection
        var subscription = new Subscription(subscriberId, topic, internalCallBack, catchAll);
        _subscriptions.Add(subscription);
    }

1 个答案:

答案 0 :(得分:1)

您应该使用BeginInvoke,因为Invoke会导致死锁。 Invoke是一种锁,所以它会等待,阻止当前线程直到动作结束。 BeginInvoke将请求放在 Windows消息泵上,稍后UI线程将处理该请求,而不会阻止工作线程。

即使Windows GUI不是多线程的,您仍然可以使其死锁。查看下面的文章。

更多