利用委托实现异步调用

使用BackgroundWorker组件 一文中,阐述了在Winform编程中,使用BackgroundWorker组件实现异步调用,本文主要讲述利用委托实现异步。

以下描述摘抄于MSDN:

异步委托提供以异步方式调用同步方法 的能力。

当同步调用委托时,Invoke()方法直接对当前线程调用目标方法;

当异步调用委托时,CLR将对请求进行排 队并立即返回到调用方,将对来自线程池的线程调用该目标方法,提交请求的原始线程继续与目标方法并行执行,该目标方法是对线程池线程运行的.

1)、 BeginInvoke()方法

BeginInvoke()方法启动异步调用,它与需要异步执行的方法具有相同的参 数。

另外,还有两个可选参数:第一个参数是AsyncCallback委托,该委托引用在异步调用完成时要调用的方法;第二个参数是用户 定义的对象,该对象可向回调方法传递信息;

BeginInvoke立即返回,不等待异步调用完成;

BeginInvoke 返回IAsyncResult,这个结果可用于监视异步调用的进度;

2)、EndInvoke()方法

EndInvoke()方法检索异步调用的结果;

在调用BeginInvoke()方法后,可以随时调用EndInvoke() 方法,如果异步调用尚未完成,则EndInvoke()方法将一直阻止调用线程,直到异步调用完成后才允许调用线程执行;

EndInvoke() 的参数需要异步执行的方法的out和ref参数以及由BeginInvoke()返回的IAsyncResult。

 

下 面通过代码阐述异步委托:

代码一,同步执行:
 public delegate int MathDelegate(int x);  
public class MathClass
 { public int Add(int x)
  { Thread.Sleep(10000);//此处模拟长时间执行的任务 r
    eturn x + x; }
 }
 public class Program 
 
  public static void Main(string[] args)
   {
   MathClass addClass = new MathClass(); 
MathDelegate mathDel = new MathDelegate(addClass.Add); //同步执行
 int syncResult = mathDel(8);
 Console.WriteLine("Sync Proccessing operation...");//这一行只有SyncMethod完成以后才能显示
 Console.WriteLine("Sync Result is: {0}", syncResult); 
Console.ReadLine();
}
 }
当程序执行到 <span style="color: blue;">int </span>syncResult = mathDel(8); 的时候,主线程将等待至少10秒的时间(Add方法的执行),才能执行
后面的代码,也即在期间,应用程序没有响应,不能执行其他的任何操作,直到Add方法返回结果。
代码二,异步执行:

我们稍微修改一下Main的代码:

<span style="color: blue;">public static void </span>Main(<span style="color: blue;">string</span>[] args)<br />{<br />    <span style="color: rgb(43, 145, 175);">MathClass </span>addClass = <span style="color: blue;">new </span><span style="color: rgb(43, 145, 175);">MathClass</span>();<br />    <span style="color: rgb(43, 145, 175);">MathDelegate </span>mathDel = <span style="color: blue;">new </span><span style="color: rgb(43, 145, 175);">MathDelegate</span>(addClass.Add);<br /><br />    <span style="color: rgb(43, 145, 175);">IAsyncResult </span>async = mathDel.BeginInvoke(9, <span style="color: blue;">null</span>, <span style="color: blue;">null</span>);<span style="color: green;">//在另外的线程里,调用Add方法<br />    </span><span style="color: rgb(43, 145, 175);">Console</span>.WriteLine(<span style="color: rgb(163, 21, 21);">"Async Proccessing operation..."</span>);<span style="color: green;">//立即打印到终端设备<br />    </span><span style="color: blue;">int </span>asyncReuslt = mathDel.EndInvoke(async);<br />    <span style="color: rgb(43, 145, 175);">Console</span>.WriteLine(<span style="color: rgb(163, 21, 21);">"Result is: {0}"</span>, asyncReuslt);<br /><br />    <span style="color: rgb(43, 145, 175);">Console</span>.ReadLine();<br />}

在这段代码中,在开始并没有直接调用方法,而是使用BeginInvoke()方法,返回IAsyncResult 对象。

代码三,IsCompleted,轮询异步调用完成
使用IAsyncResult实例的IsCompleted属性,以获取异步操作是否已完成的指示,如果操作完成则为True,否则为 False。

修改一下Main的代码:

<span style="color: blue;">public static void </span>Main(<span style="color: blue;">string</span>[] args)<br />{<br />    <span style="color: rgb(43, 145, 175);">MathClass </span>addClass = <span style="color: blue;">new </span><span style="color: rgb(43, 145, 175);">MathClass</span>();<br />    <span style="color: rgb(43, 145, 175);">MathDelegate </span>mathDel = <span style="color: blue;">new </span><span style="color: rgb(43, 145, 175);">MathDelegate</span>(addClass.Add);<br /><br />    <span style="color: rgb(43, 145, 175);">IAsyncResult </span>async = mathDel.BeginInvoke(9, <span style="color: blue;">null</span>, <span style="color: blue;">null</span>);<span style="color: green;">//在另外的线程里,调用Add方法<br />    </span><span style="color: rgb(43, 145, 175);">Console</span>.WriteLine(<span style="color: rgb(163, 21, 21);">"Async Proccessing operation..."</span>);<span style="color: green;">//立即打印到终端设备<br /><br />    </span><span style="color: blue;">int </span>i = 1;<br />    <span style="color: blue;">while </span>(async.IsCompleted==<span style="color: blue;">false</span>)<br />    {<br />        <span style="color: rgb(43, 145, 175);">Thread</span>.Sleep(i * 1000);<br />        <span style="color: rgb(43, 145, 175);">Console</span>.WriteLine(<span style="color: rgb(163, 21, 21);">"IsCompleted:{0},{1}"</span>, async.IsCompleted, i);<br />        i++;<br />    }<br />    <span style="color: blue;">int </span>asyncReuslt = mathDel.EndInvoke(async);<br />    <span style="color: rgb(43, 145, 175);">Console</span>.WriteLine(<span style="color: rgb(163, 21, 21);">"Result is: {0}"</span>, asyncReuslt);<br /><br />    <span style="color: rgb(43, 145, 175);">Console</span>.ReadLine();<br />}
代码四,AsyncCallback,异步调用完成时执行回调方法

如果启动异步调用的线程不需要是处理结果的线程,则可以在调用完成时执行回调方法;

如果要使用回调方法,必须将引用回调方法AsyncCallback委托传递给BeginInvoke()方法,也可以传递包含回调方法将要使用的 信息的对象。

修改一下Main的代码:

<span style="color: blue;">public static void </span>Main(<span style="color: blue;">string</span>[] args)<br />{<br />    <span style="color: rgb(43, 145, 175);">MathClass </span>addClass = <span style="color: blue;">new </span><span style="color: rgb(43, 145, 175);">MathClass</span>();<br />    <span style="color: rgb(43, 145, 175);">MathDelegate </span>mathDel = <span style="color: blue;">new </span><span style="color: rgb(43, 145, 175);">MathDelegate</span>(addClass.Add);<br /><br />    <span style="color: rgb(43, 145, 175);">IAsyncResult </span>async = mathDel.BeginInvoke(9, <span style="color: blue;">new </span><span style="color: rgb(43, 145, 175);">AsyncCallback</span>(CompleteMethod), <span style="color: rgb(163, 21, 21);">"信息来自于主线程"</span>);<span style="color: green;">//在另外的线程里,调用Add方法<br />     </span><span style="color: rgb(43, 145, 175);">Console</span>.WriteLine(<span style="color: rgb(163, 21, 21);">"Async Proccessing operation..."</span>);<span style="color: green;">//立即打印到终端设备<br />   </span><br />    <span style="color: rgb(43, 145, 175);">Console</span>.ReadLine();<br />}<br /><span style="color: blue;">private static void </span>CompleteMethod(<span style="color: rgb(43, 145, 175);">IAsyncResult </span>async)<br />{<br />    <span style="color: rgb(43, 145, 175);">AsyncResult </span>ar = (<span style="color: rgb(43, 145, 175);">AsyncResult</span>)async;<br /><br />    <span style="color: rgb(43, 145, 175);">MathDelegate </span>del = (<span style="color: rgb(43, 145, 175);">MathDelegate</span>)ar.AsyncDelegate;<br />    <span style="color: blue;">int </span>result = del.EndInvoke(async);<br /><br />    <span style="color: blue;">string </span>mainTheadMsg = ar.AsyncState <span style="color: blue;">as string</span>;<br />    <span style="color: rgb(43, 145, 175);">Console</span>.WriteLine(<span style="color: rgb(163, 21, 21);">"{0}, Result is: {1}"</span>, mainTheadMsg, result);<br />}
Add方法调用完成以后,调用 CompleteMethod 方法
结束



pixy.gif?x-id=b7583501-518a-88a9-a91d-662e0334b0c3

转载于:https://www.cnblogs.com/kevinzhwl/archive/2010/06/12/1756875.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值