成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

WCF異步調用技巧掌握

開發 開發工具
WCF異步調用的實現方法是一個比較復雜的步驟。在這里我們為大家詳細介紹了有關WCF異步調用的相關技巧,希望對大家有所幫助。

WCF是一款功能強大的開發框架,可以幫助我們實現跨平臺的互聯網解決方案。在這里我們可以學習到有關WCF異步調用的一些實現方法。#t#

WCF與Web Service不同的是,當我們定義了服務契約的操作時,不管是通過ChannelFactory創建服務代理對象,還是通過SvcUtil的默認方式生 成服務代理對象,客戶端在調用這些代理對象時,都無法直接實現WCF異步調用。例如,對于如下的服務操作定義:

 

  1. [OperationContract]  
  2. Stream TransferDocument(Document document); 

 

在調用代理對象的方法時,我們無法找到對應于TransferDocument()操作的BeginTransferDocument()和EndTransferDocument()異步方法。

這 樣的設計使得我們無法通過編程方式WCF異步調用服務的操作,除非我們在定義服務接口時,直接加入相關操作的異步方法。然而,這又直接導致了服務的設計與方法 調用方式之間的耦合

。一個好的框架設計要素在于,不管客戶端的調用方式(同步或者異步),服務的設計與實現應該是一致的。對于服務的設計者而言,在設計之 初,就不應該去考慮服務的調用者調用的方式。換言之,服務操作究竟是否采用異步方式,應該由客戶端的調用者決定。因此,所有與WCF異步調用相關的內容應該只與 客戶端相關。WCF遵循了這一規則。

在我編寫的應用程序中,會暴露一個傳送文檔文件的服務操作。我并不知道也并不關心調用該操作的客戶端是否采用異步方式。因此,如上所述的服務操作定義是完全正確的。

那 么,客戶端究竟應該如何執行異步調用呢?如果采用編程方式獲得服務代理對象,這一問題會變得比較糟糕。因為我將服務契約的定義單獨形成了一個程序集,并在 客戶端直接引用了它。然而,在這樣的服務契約程序集中,是沒有包含異步方法的定義的。因此,我需要修改在客戶端的服務定義,增加操作的異步方法。這無疑為 服務契約的重用帶來障礙。至少,我們需要在客戶端維持一份具有異步方法的服務契約。

所幸,在客戶端決定采用WCF異步調用我所設計的服務操作時,雖然需要修改客戶端的服務契約接口,但并不會影響服務端的契約定義。因此,服務端的契約定義可以保持不變,而在客戶端則修改接口定義如下:

 

  1. [ServiceContract]  
  2. public interface IDocumentsExplorerService  
  3. {  
  4. [OperationContract]  
  5. Stream TransferDocument(Document document);  
  6. [OperationContract(AsyncPattern = true)]  
  7. IAsyncResult BeginTransferDocument(Document document,  
  8. AsyncCallback callback, object asyncState);  
  9. Stream EndTransferDocument(IAsyncResult result);  
  10. }  

 

注意,在BeginTransferDocument()方法上,必須在OperationContractAttribute中將AsyncPattern屬性值設置為true,因為它的默認值為false。

調用方式如下:

  1. BasicHttpBinding binding = 
    new BasicHttpBinding();  
  2. binding.SendTimeout =
     
    TimeSpan.FromMinutes(10);  
  3. binding.TransferMode =
     TransferMode.Streamed;  
  4. binding.MaxReceivedMessageSize = 
    9223372036854775807;  
  5. EndpointAddress address = 
    new EndpointAddress  
  6. ("http://localhost:8008/Document
    ExplorerService");  
  7. ChannelFactory<IDocuments
    ExplorerService
    > factory =   
  8. new ChannelFactory<IDocuments
    ExplorerService
    >(binding,address);  
  9. m_service = factory.CreateChannel();  
  10.  
  11. ……  
  12. IAsyncResult result = m_service.
    BeginTransferDocument(doc,null,null);  
  13. result.AsyncWaitHandle.WaitOne();  
  14. Stream stream = m_service.
    EndTransferDocument(result); 

如果采用SvcUtil生成客戶端代理文件,可以有更好的方式實現異步,也就是使用SvcUtil的/async開關,例如:
svcutil /async http://localhost:8008/DocumentExplorerService
唯一不足的是,它會不分青紅皂白,為所有服務操作都生成對應的異步方法。這樣的做法未免過于武斷。

合理地利用服務的WCF異步調用,可以有效地提高系統性能,合理分配任務的執行。特別對于UI應用程序而言,可以提高UI的響應速度,改善用戶體驗。在我編寫的應用程序中,下載的文件如果很大,就有必要采用異步方式。

對于異步調用的完成,雖然WCF提供了諸如阻塞、等待和輪詢等機制,但***的方式還是使用回調。也就是利用Begin方法參數中的AsyncCallback對象。這是一個委托對象,它的定義如下所示:
public delegate void AsyncCallback(IAsyncResult ar);
利用異步方式執行服務操作,使得服務在執行過程中不會阻塞主線程,當方法執行完成后,通過AsyncCallback回調對應的方法,可以通知客戶端服務執行完畢。例如:

 

  1. //Invoke it Asynchronously  
  2. m_service.BeginTransferDocument
    (m_doc,OnTransferCompleted,null);  
  3. //Do some work;  
  4. //callback method  
  5. void OnTransferCompleted
    (IAsyncResult result)  
  6. {  
  7. Stream stream = m_service.
    EndTransferDocument(result);  
  8. result.AsyncWaitHandle.Close();  
  9. lbMessage.Text = string.Format
    ("The file {0} had been 
    transfered sucessfully.",   
  10. m_doc.FileName);  

 

在調用BeginTransferDocument()方法之后,主線程不會被阻塞,仍然可以繼續執行其它工作。而當服務方法執行完畢之后,會自動調用回調方法,執行方法中的內容。

上 述實現存在一個問題,就是對于lbMessage控件的操作。由于回調方法并非運行在主線程中,如果回調方法需要更新與WCF異步調用結果相關的界面,例如本例 中的lbMessage控件,則需要將回調的調用封送(Marshal)到當前主程序界面的同步上下文中。我們可以使用 SynchronizationContext以及它的SendOrPostCallback委托,對調用進行封送:

  1. public ExplorerClientForm()  
  2. {  
  3. InitializeComponent();  
  4. m_synchronizationContext = 
    SynchronizationContext.Current;  
  5. }  
  6. private SynchronizationContext
     m_synchronizationContext; 

 

則回調方法修改為:

  1. //callback method  
  2. void OnTransferCompleted
    (IAsyncResult result)  
  3. {  
  4. Stream stream = m_service.
    EndTransferDocument(result);  
  5. result.AsyncWaitHandle.Close();  
  6.  
  7. SendOrPostCallback callback = delegate 
  8. {  
  9. lbMessage.Text = string.Format
    ("The file {0} had been transfered 
    sucessfully.",   
  10. m_doc.FileName);  
  11. };  
  12. m_synchronizationContext.
    Send(callback,null);  

在WCF異步調用方法時,由于對BeginTransferDocument()和EndTransferDocument()方法的調用可能會在不同的方法體中,因而我將服務代理對象定義為private字段。如果希望將服務對象定義為一個局部變量,可以在調用BeginTransferDocument()方法時,將代理對象傳遞到方法的asyncState參數中,然后在調用EndTransferDocument()方法之前,通過IAsyncResult獲得準確的服務代理對象:
m_service.BeginTransferDocument(m_doc,OnTransferCompleted,m_service);
將m_service作為asyncState對象傳入之后,在調用EndTransferDocument()方法之前,就可以根據它先獲得服務代理對象

 

  1. IDocumentsExplorerService m_service = 
    result.AsyncState as IDocuments
    ExplorerService;  
  2. Stream stream = m_service.
    EndTransferDocument(result);  
  3. //rest codes 

以上就是WCF異步調用的相關實現方法。

責任編輯:曹凱 來源: itstrike.cn
相關推薦

2010-02-22 13:28:05

WCF異步調用

2009-11-09 10:50:30

WCF異步調用

2009-11-06 15:54:15

WCF異步調用

2009-12-07 14:35:42

WCF異步調用

2009-12-07 14:26:47

WCF異步調用

2010-02-25 09:13:34

WCF異步調用

2009-11-09 15:49:01

WCF異步調用

2010-03-01 14:01:50

WCF服務異步調用

2009-12-22 18:43:00

WCF異步調用

2009-12-23 11:24:42

WCF附加屬性

2010-02-22 17:58:06

WCF異步上傳

2009-10-20 16:48:30

C#委托

2009-07-01 13:58:00

JavaScript異

2009-12-21 15:53:56

WCF獲取客戶端IP

2010-01-11 17:24:19

VB.NET異步調用

2011-03-02 08:57:22

jQueryJavaScript

2012-10-29 10:59:27

Windows 8

2021-03-29 09:26:44

SpringBoot異步調用@Async

2009-08-21 11:24:16

C#異步調用

2009-07-01 14:05:23

JavaScript異
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲日日 | 日韩欧美国产精品一区二区三区 | 国产伦精品一区二区三区照片91 | 日韩精品成人在线 | 国产小视频精品 | 午夜在线视频一区二区三区 | 国产一区二 | 日韩精品一区二 | 中文字幕在线一区二区三区 | 国产999在线观看 | 国产一区欧美 | 99在线国产 | 欧美激情久久久久久 | 久久97精品| 国产美女精品视频 | 久久亚洲欧美日韩精品专区 | 黄a在线播放 | 欧美激情精品久久久久久变态 | 91在线观看网址 | 国产在线观看一区二区三区 | 99精品久久久久 | 亚洲视频一区二区三区 | 久久国产免费 | 亚洲乱码一区二区三区在线观看 | 黄免费观看 | 久久电影一区 | 美女视频一区二区三区 | 欧美日韩不卡 | 国内自拍视频在线观看 | 久久国产精品色av免费观看 | 国产伦精品一区二区三区照片91 | 午夜av电影 | 黄网站免费在线 | 成人欧美一区二区三区在线播放 | 久操av在线| 久综合 | 久草视 | 欧美久久一级 | 亚洲色片网站 | 亚洲自拍偷拍欧美 | 日韩av成人在线 |