本篇学习了基于事件的异步模式。
问题的提出:基于IAsyncResult接口的异步模式.在异步回调中,回调线程不同于调用线程。使用Windows窗体或WPF时,这就是个问题,因为Windows窗体和WPF控件绑定到一个线程上。对于每个控件,都只能从创建该控件的线程中调用方法,也就是说后台线程不能直接在这个线程中访问UI控件,解决办法如下:
(1):在Windows窗体中,唯一可以从非创建线程中调用的方法是Invoke()、BeginInvoke()、EndInvoke()和属性InvokeRequired.
BeginInvoke和EndInvoke()是Invoke()的异步版本。这些方法会切换到创建控件的线程上,调用赋予一个委托参数的方法,该委托
参数可以传递给这些方法。这些方法的使用并不简单。我曾经在写串口操作时就碰到了问题,当时没理解正在的原因。当时把这个问题做了一个笔记:线程间操作无效: 从不是创建控件" XX" 的线程访问它
(2):基于事件的异步模式----是.NET2.0新组件和新异步模式一起开发的原因。在基于事件的异步模式中,异步组件提供了后缀为Asyc的方法。
例如:DoATask()异步版本是DoATaskAsync(),为了得到结果信息还需要定义一个后缀为Completed的事件,例如:DoATaskCompleted。
DoATaskAsync()方法中的操作在后台线程运行时,会在调用线程中触发DoATaskCompleted事件.在基于事件的异步模式中,异步组件可以
支持取消操作,提供进度信息.对于取消操作,方法的名称应是CancelAsync()。对于进度信息,应提供后缀为ProgressChanged的事件
DoATaskProgressChanged.
实例讲解:BackgroundWorker类(是异步事件模式的一种解决方案),另一实现异步事件模式类是System.Net命名空间中的组件WebClient(WebRequest和WebResponse类,基于异步模式和IAsyncResult接口)。
要明白BackgroundWorker类的事件引发关系
(1)方法RunWorkerAsync()引发DoWork事件,在一个单独的线程中启动异步任务
(2)无论消息与否,只要完成异步任务,就引发RunWorkerCompleted事件
(一)代码:
(1)EventAsync.Designer.cs
(2)EventAsync.cs
(二)运行截图: