自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

WCF中的異步實(shí)現(xiàn)

開(kāi)發(fā) 后端
對(duì)于WCF中通訊的雙方來(lái)說(shuō),客戶端可以異步的調(diào)用服務(wù);服務(wù)端對(duì)服務(wù)也能以異步的方式實(shí)現(xiàn)。這就涉及到兩個(gè)方面:WCF客戶端異步調(diào)用服務(wù);服務(wù)端的異步實(shí)現(xiàn)本節(jié)是Artech《WCF技術(shù)剖析(卷1)》一篇讀書(shū)筆記。在文章中老A介紹了客戶端如何以異步的方式消費(fèi)WCF服務(wù),以及服務(wù)的異步實(shí)現(xiàn),個(gè)人對(duì)這幾種方式做個(gè)總結(jié)。

目錄:

1.WCF客戶端異步調(diào)用服務(wù)

2.服務(wù)端的異步實(shí)現(xiàn)

WCF客戶端異步調(diào)用服務(wù)主要通過(guò)生成異步的代理類,然后調(diào)用其中的異步方法來(lái)實(shí)現(xiàn)異步調(diào)用。

異步代理類的生成:

通過(guò)SvcUtil /async 直接生產(chǎn)異步代理;

通過(guò)添加應(yīng)用的方式,點(diǎn)擊”添加引用“的“高級(jí)”按鈕,在彈出來(lái)的對(duì)話框中選擇“生成異步”。如圖:

生成的異步調(diào)用代理類部分借口:

  1. [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Advanced)]  
  2. public System.IAsyncResult BeginAdd(int x, int y, System.AsyncCallback callback, object asyncState) {  
  3. return base.Channel.BeginAdd(x, y, callback, asyncState);  
  4. }  
  5.  
  6.  
  7. public void AddAsync(int x, int y) {  
  8. this.AddAsync(x, y, null);  
  9. }  
  10.  
  11. public void AddAsync(int x, int y, object userState) {  
  12. if ((this.onBeginAddDelegate == null)) {  
  13. this.onBeginAddDelegate = new BeginOperationDelegate(this.OnBeginAdd);  
  14. }  
  15. if ((this.onEndAddDelegate == null)) {  
  16. this.onEndAddDelegate = new EndOperationDelegate(this.OnEndAdd);  
  17. }  
  18. if ((this.onAddCompletedDelegate == null)) {  
  19. this.onAddCompletedDelegate = new System.Threading.SendOrPostCallback(this.OnAddCompleted);  
  20. }  
  21. base.InvokeAsync(this.onBeginAddDelegate, new object[] {  
  22. x,  
  23. y}, this.onEndAddDelegate, this.onAddCompletedDelegate, userState);  
1、客戶端異步調(diào)用服務(wù)

客戶端異步調(diào)用服務(wù)主要方式有幾種:

1.1、直接調(diào)用異步方法:

在生成的代理類中,有BeginAdd\EndAdd等服務(wù)契約中定義的Add操作的異步實(shí)現(xiàn)。直接調(diào)用BeginAdd方法,實(shí)現(xiàn)客戶端異步調(diào)用服務(wù)端方法。在調(diào)用BeginAdd方法后,可執(zhí)行一些其他操作,這些操作與服務(wù)端Add調(diào)用并行執(zhí)行,Add調(diào)用最終通過(guò)EndAdd方法得到。

測(cè)試代碼如下:

客戶端代碼:

  1. IAsyncResult asyncResult = calculatorClient.BeginAdd(1, for (int i = 0; i < 10; i++)   
  2. {  
  3. Console.WriteLine(i);  
  4. }  
  5. int resul = calculatorClient.EndAdd(asyncResult);  
  6. Console.WriteLine(string.Format("計(jì)算結(jié)果:{0}",resul)); 
服務(wù)端代碼:
 
  1. public int Add(int x, int y)   
  2. {  
  3. for (int i = 0; i < 20; i++)  
  4. {  
  5. Console.WriteLine(i);  
  6. }  
  7. Console.WriteLine("開(kāi)始計(jì)算...");  
  8. return x + y;  
 

服務(wù)端輸出如下:

客戶端輸出如下:

這種方式使用了EndAdd方法,如果服務(wù)端沒(méi)有執(zhí)行完成,當(dāng)前線程會(huì)被阻塞直到異步調(diào)用的服務(wù)完成后結(jié)束。如客戶端代碼保持不變,將服務(wù)實(shí)現(xiàn)改為如下:

  1. for (int i = 0; i < 20; i++)   
  2. {  
  3. Console.WriteLine(i);  
  4. }  
  5. Thread.Sleep(5000);  
  6. Console.WriteLine("開(kāi)始計(jì)算...");  
  7. return x + y; 

如果在服務(wù)端讓線程睡眠幾秒,就可看到客戶端會(huì)被阻塞:

1.2、通過(guò)回調(diào)的方式異步調(diào)用服務(wù):

在生成的異步調(diào)用代理類中,還可以通過(guò)回調(diào)用服務(wù):

  1. IAsyncResult asyncResult = calculatorClient.BeginAdd(1, 2,  
  2. delegate(IAsyncResult asyncResult)  
  3. {  
  4. int [] array = asyncResult.AsyncState as int [];  
  5. int result= calculatorClient.EndAdd(asyncResult1);  
  6. calculatorClient.close()  
  7. Console.WriteLine(string.Format("{0}+{1}={2}", array[0], array[1], result));  
  8. }, new []{1,2}); 

這種方式是對(duì)服務(wù)的異步調(diào)用完成以后,自動(dòng)調(diào)用回調(diào)來(lái)獲取結(jié)果。

1.3、通過(guò)為異步操作注冊(cè)事件

  1. //進(jìn)行異步調(diào)用  
  2. calculatorClient.AddAsync(10, 36, new[] { 1000 });  
  3. //為異步調(diào)用完成定義觸發(fā)事件  
  4. calculatorClient.AddCompleted += calculatorClient_AddCompleted;  
  5. Console.WriteLine("服務(wù)調(diào)用完成...");  
  6. Console.ReadKey();  
  7. //異步調(diào)用完成后執(zhí)行  
  8. privatestaticvoid calculatorClient_AddCompleted(object obj, AddCompletedEventArgs args)  
  9. {  
  10. var array = args.UserState as int[];  
  11. int result = args.Result;  
  12. Console.WriteLine(result);  

2、服務(wù)的異步實(shí)現(xiàn):

將服務(wù)實(shí)現(xiàn)定義成異步的方式,需要將OperationContract的AsyncPattern設(shè)置為true;應(yīng)用到BeginXX接口上,并且此操作的最后兩個(gè)參數(shù)必須為AsyncCallback ,object;需要有一個(gè)EndXX(IAsyncResult asyncResult)的接口與異步調(diào)用的接口匹配。需要注意的是EndXX(IAsyncResult asyncResult)不能再次聲明為契約接口,也就是不能再標(biāo)記為OperationContract。

將服務(wù)定義為異步服務(wù),契約定義如下 :

  1. [OperationContract(AsyncPattern = true)]   
  2. IAsyncResult BeginCalculator   
  3. (int x,int y ,AsyncCallback asyncCallback, object state);  
  4. void EndCalculator(IAsyncResult); 

然后在實(shí)現(xiàn)契約接口的服務(wù)中,將方法實(shí)現(xiàn)為異步的。

將契約接口聲明為單向,也就是OneWay,這樣客戶端對(duì)此服務(wù)接口調(diào)用就是異步的。因?yàn)樗鼰o(wú)需等待服務(wù)端返回,客戶端只需將消息發(fā)送到傳輸層就立即返回。

那能不能將將這種客戶端對(duì)服務(wù)端的異步調(diào)用直接標(biāo)記為單向的,那是不是可以不用生成異步代理類(即上述SvcUtil /Ayncs或者在添加引用時(shí)聲明將代理類生成異步操作)?答案是否定的,因?yàn)镺neWay要求方法的返回值為void,而異步的方法需要IAsyncResult最為返回值,這兩者矛盾的。

責(zé)任編輯:彭凡 來(lái)源: 博客園
相關(guān)推薦

2010-02-25 09:13:34

WCF異步調(diào)用

2010-03-01 14:01:50

WCF服務(wù)異步調(diào)用

2009-12-21 14:10:26

WCF異步調(diào)用

2009-11-09 10:50:30

WCF異步調(diào)用

2010-02-22 17:58:06

WCF異步上傳

2009-03-13 15:24:50

catchDisposeWCF

2011-02-24 12:53:51

.NET異步傳統(tǒng)

2010-03-01 10:12:54

WCF異步操作

2009-11-06 15:54:15

WCF異步調(diào)用

2009-12-22 18:43:00

WCF異步調(diào)用

2010-03-01 15:08:05

WCF單向操作

2009-12-07 14:26:47

WCF異步調(diào)用

2009-12-07 14:35:42

WCF異步調(diào)用

2010-03-01 10:26:40

WCF異步服務(wù)

2010-02-24 09:59:19

WCF服務(wù)異步代理

2010-02-22 13:28:05

WCF異步調(diào)用

2009-12-21 14:58:57

WCF用戶密碼認(rèn)證

2009-11-09 15:49:01

WCF異步調(diào)用

2009-11-09 15:58:07

WCF回調(diào)方法

2009-12-21 10:00:46

WCF基礎(chǔ)開(kāi)發(fā)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)