来说明如何处理后台线程通过异步方式来更新UI。他的方案非常棒,但是客户端稍稍复杂了一点,在非常复杂的场景可能会发生问题。我在实际工作中遇到这个问题的时候不是以异步委托的面目出现的,而是以接口方式实现的,这样的情况更具有一般性。我通常是在若干个订阅端(Subscriber)实现这个“订阅主题”的接口,在一个发布端(Publisher)实现同样一个接口,并提供对订阅者的管理。事实上,在业务层实现这个发布端的接口是容易的,但在GUI的订阅端有可能会发生另外一些问题,例如如果业务端与GUI客户端并不在同一个线程中。典型的例子就是通常将“长任务”放到某个后台线程中去执行,在执行过程中可以在任何时候将执行状态报告给前台线程。显然,这是一个处理一般性Observe模式的通用方案。

实现的方法有点变态,和Mixin的原理很相似。在一个AsyncObjectBase的基类或者该基类的派生类上动态实现客户端接口,同时实现对订阅者的管理。之所以动态化是为了适应客户端接口中的任何签名的方法。

以下是关于AsyncObjectBase的简单的API:

提供了两个方法来管理订阅者:

Register(object):添加订阅者。

Unregister(object):删除订阅者。

提供了三个方法来管理发布状态:

Disable():关闭发布。

Enable(object):仅开放指定的订阅者。

Enable():开放所有的订阅者。

提供了两个静态方法来实例化接口的服务端实现:

AsyncObjectBase GetObject(Type, Type):获取指定基类、指定接口类型的服务端实现(指定基类必须继承自AsyncObjectBase)。

AsyncObjectBase GetObject(Type):获取指定接口类型的服务端实现。

从AsyncObjectBase派生一个自定义类型来重写委托指派方法可实现复杂的委托(不仅限于基于System.Windows.Form.Control的实现类)。这三个方法是:

void CheckDelegate(Delegate, object[], out bool);

object ChecjDelegate2(Delegate, object[], out bool);

以上两个方法都是捕获委托的,差异是前者实现的委托无返回值,而后者实现的委托有返回值。最后一个参数通知服务器该委托是否已经被捕获,以决定是否需要继续raise这个委托。这是本方案加入自定义委托捕获的唯一途径。

bool GetEnabled(object):返回是否需要向指定委托者发布。

示例实现这样一个场景:后台线程打开一个文本文件,然后一段一段地向多个订阅者发布。

有两点需要说明:

一、示例未实现多接口,当然如果需要的话,改造过程非常简单。

二、未处理带返回值的方法的接口,当然如果需要的话,改造过程也非常简单。