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

基于事件的C#異步編程模式淺析

開發 后端
基于事件的C#異步編程模式向你介紹了區別于IAsyncResult模式的另外一種C#異步編程模式,希望對你了解和學習C#異步編程模式有所幫助。

基于事件的C#異步編程模式是什么呢?我們現在開始向你慢慢介紹:

基于事件的C#異步編程模式是比IAsyncResult模式更高級的一種異步編程模式,也被用在更多的場合。對于相對簡單的應用程序可以直接用 .Net 2.0 新增的 BackgroundWorker 組件來很方便的實現,對于更復雜的異步應用程序則需要自己實現一個符合基于事件的C#異步編程模式的類。這兩者對我都是新東西,先從簡單的入手,下一篇里我再去嘗試復雜類模型的實現

基于事件的C#異步編程模式概述

支持基于事件的C#異步編程模式的類會有若干個 MethodNameAsync 方法表示開始異步操作,并有對應的 MethodNameCompleted 事件。類里面還可能會有 CancelAsync 或 MethodNameAsyncCancel 方法用于取消異步操作,并可以有  ProgressChanged 或 MethodNameProgressChanged 事件來跟蹤執行進度。下面分別作一下解釋

MethodNameAsync 方法可以有兩個重載:單調用和多調用,多調用有一個額外的狀態對象參數  userState。userState 參數用來區分各次異步操作,使得我們可以多次調用多調用形式的方法而不需要等待任何異步操作的完成(在學習 IAsyncResult 模式時我把狀態對象僅僅當成傳給回調方法的一個條件來用,可能在使用模式時這么做并沒有什么關系,但在實現模式時不把狀態對象用作異常調用的唯一標識而另作他用就值得商榷了)。而單調用形式的方法如果在前一個調用尚未完成時調用將會拋出 InvalidOperationException 異常

如果有多個異步方法,則應使用 CancelAsync 方法來取消掛起的操作,并可使用 userState 來取消指定的掛起任務。如果只有一個異步方法則可以使用 MethodNameAsyncCancel 方法

另外 MSDN 上說:一次只支持一個掛起的操作的方法(如 Method1Async(string param) )是不可取消的。這句話我還沒有理解,不可能說是單調用的異步方法就不能取消吧,BackgroundWorker 上都是這樣做的

先不管了,接著看ProgressChanged 事件。它有一個 ProgressChangedEventArgs 參數,事件處理程序通過檢查該參數的 ProgressPercentage 屬性來獲取任務完成的百分比。如果有多個異步操作掛起,也可以通過檢查參數的 UserState 屬性來分辨操作。如果需要用 ProgressChanged 事件來報告增量結果,則可以把結果保存在派生自 ProgressChangedEventArgs 的類中,并在事件處理程序中使用

基于事件的C#異步編程模式之BackgroundWorker

BackgroundWorker 很好的符合了事件異步操作模式。它有兩個重載版本的 RunWorkerAsync 方法(均為單調用形式)和 RunWorkerCompleted 事件,并有 CancelAsync 方法以及 ProcessChanged 事件。不同的是 BackgroundWorker 增加了 DoWork 事件,在 RunWorkerAsync 方法調用時發生,以達到將實際執行的開始方法與 BackgroundWorker 分離的目的。還需要提一下的是 WorkerReportsProcess 屬性和 ReportProcess 方法,前者指示能否報告進度更新,后者引發 ProcessChanged 事件,它們會在接下來的 Demo 里用到

基于事件的C#異步編程模式的實例應用:

因為平時經常要處理幾十兆的文本文件,這個 Demo 就做一個讀取文件并顯示進度的控制臺程序。先看類名和字段

  1. class BackgroundWorkerDemo  
  2. {  
  3.   private BackgroundWorker m_bw;  
  4.   string m_FilePath;  

構造函數接收文件路徑為參數,設置文件路徑并初始化 BackgroundWorker

  1. public BackgroundWorkerDemo(string filePath)  
  2.   {  
  3. m_FilePath = filePath;  
  4.  
  5. m_bw = new BackgroundWorker();  
  6. m_bw.WorkerReportsProgress = true;  
  7. m_bw.DoWork += new DoWorkEventHandler(  
  8.  
  9. BackgroundWorker_DoWork);  
  10. m_bw.ProgressChanged +=   
  11.  
  12. new ProgressChangedEventHandler(  
  13.  
  14. BackgroundWorker_ProgressChanged);  
  15. m_bw.RunWorkerCompleted +=   
  16.  
  17. new RunWorkerCompletedEventHandler(  
  18.  
  19. BackgroundWorker_RunWorkerCompleted);  
  20.   }  

接下來看這三個事件的處理程序。每一個事件都有各自的 EventArgs 參數類型,都很簡單就不多說了

***個 BackgroundWorker_DoWork 方法寫得我有些郁悶。我在方法里取文件長度,先是直接取 StreamReader.BaseStream.Length 或 FileInfo.Length ,結果卻導致很多文件讀不到 100% 就結束了,不得已改成先把整個文件讀一次得到字符串的長度。這樣的方法當然性能不好了,主要是因為自己對 IO 一直就不夠清楚,等下一個主題重新認識下 IO 再回頭過來改吧。也望有經驗的朋友賜教,感激不盡

  1.   /**//// ﹤summary﹥  
  2.   /// DoWork event process method  
  3.   /// ﹤/summary﹥  
  4.   /// ﹤param name="sender"﹥﹤/param﹥  
  5.   /// ﹤param name="e"﹥﹤/param﹥  
  6.   private void BackgroundWorker_DoWork(object sender, DoWorkEventArgs e)  
  7.   {  
  8. long length;  
  9. using (StreamReader sr = new StreamReader(m_FilePath))  
  10. {  
  11. // Get file length  
  12. length = sr.ReadToEnd().Length;  
  13. }  
  14.  
  15. using (StreamReader sr = new StreamReader(m_FilePath))  
  16. {  
  17. long onePercentOfLength = length / 100;  
  18. long currentPosition = 0;  
  19. int i = 0;  
  20.  
  21. while (!sr.EndOfStream)  
  22. {  
  23.   sr.Read();  
  24.   currentPosition ++;  
  25.  
  26.   // Produce ProcessChanged event in each percent reading  
  27.   while (currentPosition ﹥ onePercentOfLength * i)  
  28.   {  
  29. ((BackgroundWorker)sender).ReportProgress(i++);  
  30.   }  
  31. }  
  32.  
  33. // e.Result will be used in RunWorkerCompleted event process method  
  34. e.Result = currentPosition;  
  35. }  
  36.   }  

基于事件的C#異步編程模式之BackgroundWorker_ProgressChanged 方法,簡單輸出當前進度

  1.   /**//// ﹤summary﹥  
  2.   /// ProgressChanged event process method  
  3.   /// ﹤/summary﹥  
  4.   /// ﹤param name="sender"﹥﹤/param﹥  
  5.   /// ﹤param name="e"﹥﹤/param﹥  
  6.   private void BackgroundWorker_ProgressChanged(  
  7. object sender, ProgressChangedEventArgs e)  
  8.   {  
  9. Console.WriteLine("Reading percents: " + e.ProgressPercentage + "%");  
  10.   } 

BackgroundWorker_RunWorkerCompleted 方法,輸出結果。這里要注意如果 RunWorkerCompletedEventArgs 參數的 Error 屬性不為空則讀取其他屬性會產生異常,然后如果 Cancelled 屬性為 true 則讀取 Result 屬性也會產生異常,因此必須依次判斷各屬性的值

  1.   /**//// ﹤summary﹥  
  2.   /// RunWorkerCompleted event process method  
  3.   /// ﹤/summary﹥  
  4.   /// ﹤param name="sender"﹥﹤/param﹥  
  5.   /// ﹤param name="e"﹥﹤/param﹥  
  6.   private void BackgroundWorker_RunWorkerCompleted(  
  7. object sender, RunWorkerCompletedEventArgs e)  
  8.   {  
  9. if (e.Error != null)  
  10. {  
  11. Console.WriteLine("Error occurs: " + e.Error.Message);  
  12. }  
  13. else if(e.Cancelled)  
  14. {  
  15. Console.WriteLine("Work cancelled");  
  16. }  
  17. else 
  18. {  
  19. Console.WriteLine("Read finished,   
  20. the file length is: " + e.Result);  
  21. }  
  22.   } 

基于事件的C#異步編程模式之向外提供一個入口方法

  1.   /**//// ﹤summary﹥  
  2.   /// Test portal  
  3.   /// ﹤/summary﹥  
  4.   public void ReadAsync()  
  5.   {  
  6. if (File.Exists(m_FilePath))  
  7. {  
  8. Console.WriteLine("Begin read");  
  9. m_bw.RunWorkerAsync();  
  10. }  
  11. else 
  12. {  
  13. throw new FileNotFoundException(  
  14. "Can't find file: " + m_FilePath);  
  15. }  
  16.   } 

***是 Main 方法,比昨天有了小小的改變,用 Console.ReadLine 代替了 Thread.Sleep 來達到阻止主線程退出的目的

  1. class BackgroundWorkerTest  
  2. {  
  3.   static void Main(string[] args)  
  4.   {  
  5. Console.Write("Input file path: ");  
  6. string filePath = Console.ReadLine();  
  7.  
  8. BackgroundWorkerDemo demo =   
  9. new BackgroundWorkerDemo(filePath);  
  10. demo.ReadAsync();  
  11.  
  12. // Thread waiting  
  13. Console.ReadLine();  
  14.   }  

基于事件的C#異步編程模式的總結

回顧一下我用委托實現 IAsyncResult 模式的 Demo ,與用 BackgroundWorker 實現的基于事件的C#異步編程模式很相似吧。而且應用程序可以通過委托的 BeginInvoke 和 EndInvoke 方法來異步執行現有的同步方法而不需要作額外的修改,BackgroundWorker 也差不多是一樣。我把這兩者看成實現對應異步操作模式的范本,在性能要求不是很高的一些異步操作場合,用好委托和 BackgroundWorker 就可以簡單有效的完成開發了。

基于事件的C#異步編程模式的基本內容就向你介紹到這里,希望對你了解和學習基于事件的C#異步編程模式有所幫助。

【編輯推薦】

  1. 概述C#加框和消框
  • 淺析C#異步操作
  • 描述C#異步Socket
  • C# Socket異步通訊實現詳解
  • C#異步編程模式IAsyncResult淺析
    1. 責任編輯:仲衡 來源: 博客園
      相關推薦

      2009-08-20 17:30:56

      C#異步編程模式

      2013-04-01 15:25:41

      異步編程異步EMP

      2009-08-17 13:34:02

      C#異步操作

      2009-08-12 15:20:21

      C#事件處理

      2009-08-21 10:17:14

      C#異步網絡編程

      2009-09-07 04:19:56

      C#窗體事件

      2009-08-21 09:20:44

      C#異步套接字

      2009-08-27 14:12:02

      C# interfac

      2009-08-13 16:27:07

      C#基于TCP協議

      2009-08-31 09:20:37

      C#事件注冊和注銷

      2009-09-09 11:29:32

      C# TextBox事

      2009-08-26 09:48:48

      C#異步套接字

      2009-03-10 13:59:41

      C#套接字編程

      2009-09-07 09:53:01

      C# DisposeDispose方法

      2009-01-16 09:58:07

      C#編程C#內存管理垃圾收集

      2009-08-20 18:47:19

      C#異步通信

      2009-04-29 09:06:18

      C#設計模式Adapter

      2013-03-08 09:33:25

      JavaScript同步異步

      2009-08-21 11:24:16

      C#異步調用

      2015-09-16 15:11:58

      C#異步編程
      點贊
      收藏

      51CTO技術棧公眾號

      主站蜘蛛池模板: 免费观看一级特黄欧美大片 | 久久国产精品一区二区三区 | 国产精品久久久久久久久久久久久 | 一本大道久久a久久精二百 国产成人免费在线 | 久久精品日 | 亚洲一区二区三区免费视频 | 久久久在线视频 | 国产日韩欧美在线观看 | 蜜桃臀av一区二区三区 | 亚洲www| 精品国产精品三级精品av网址 | 欧美一级特黄aaa大片在线观看 | 91视频三区 | 精品欧美一区二区三区久久久 | 激情欧美日韩一区二区 | 久久免费国产 | 911精品美国片911久久久 | 又爽又黄axxx片免费观看 | 欧美一级大片免费观看 | 日韩精品视频在线播放 | 国产欧美视频一区二区三区 | 国产偷录叫床高潮录音 | 欧美精品一区二区三区一线天视频 | 亚洲精品久久久久久国产精华液 | 色站综合| 国产亚洲网站 | 日韩欧美中文字幕在线视频 | 日韩久久网 | 国产偷录视频叫床高潮对白 | 亚洲精品国产成人 | 在线综合视频 | 日本网站在线看 | 99久久久久久久 | 91av在线视频观看 | 精品国产一区二区三区四区在线 | 91精品国产一区二区三区 | 免费黄色的网站 | 国产成人免费观看 | 亚洲成人av在线 | 欧美一级二级视频 | 97精品国产97久久久久久免费 |