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

.NET跨線程控件的相關(guān)操作

開發(fā) 后端
本文將為大家介紹一下.NET跨線程控件的相關(guān)操作,有時(shí)可能會(huì)出現(xiàn)一種跨線程調(diào)用的異常,這里也包括相關(guān)的解決方案。

在.NET中,如果我們在非UI線程上訪問窗體上的控件的時(shí)候,會(huì)產(chǎn)生一個(gè)跨線程調(diào)用的異常,那么如何處理這種情況呢?在上一章中,我介紹了使用Control.Invoke方法,如果你不習(xí)慣使用委托,那么.Net還為我們提供了一個(gè)組件BackgroundWorker,你可以使用這個(gè)組件,以事件的方式去處理這種跨線程的控件訪問。下面我就來詳細(xì)的介紹一下這個(gè)組件的用法。

我們先來看一下BackgroundWorker提供了哪些常用的成員,

事件

◆DoWork:我們在這個(gè)事件中,執(zhí)行需要異步處理的工作。

◆ProgressChanged:我們在這個(gè)事件中,接收并處理異步處理過程中的信息。

◆RunWorkerCompleted:我們在這個(gè)事件中,執(zhí)行異步處理結(jié)束的工作。

方法

◆RunWorkerAsync()和RunWorkerAsync(object argument):這兩個(gè)方法觸發(fā)DoWork事件,開始異步操作。

◆ReportProgress(int percentProgress)和ReportProgress(int percentProgress, object userState):這兩個(gè)方法觸發(fā)ProgressChanged事件。

◆CancelAsync:結(jié)束后臺(tái)的異步操作。

屬性

◆bool CancellationPending:指示當(dāng)前的后臺(tái)的異步操作是否正在被取消,執(zhí)行CancelAsync方法會(huì)導(dǎo)致這個(gè)屬性為true。

◆bool IsBusy:指示當(dāng)前的后臺(tái)異步操作是否正在進(jìn)行,進(jìn)行中為true。

◆bool WorkerReportsProgress:獲取或設(shè)置當(dāng)前的BackgroundWorker是否可以執(zhí)行ProgressChanged方法。

◆bool WorkerSupportsCancellation:獲取或設(shè)置當(dāng)前的BackgroundWorker是否可以執(zhí)行CancelAsync方法。

OK,有了上面這些成員,我們來看一下BackgroundWorker是如何工作的。

Step 1. 當(dāng)然是定義一個(gè)BackgroundWorker的實(shí)例,你可以從工具箱中拖拽一個(gè)BackgroundWorker控件到窗體上或者在代碼中直接聲明;

Step 2. 生成DoWork事件并在DoWork事件的中添加需要異步執(zhí)行的代碼。在異步執(zhí)行的代碼中,如果需要處理界面中的控件,請調(diào)用ReportProgress方法,而不要直接處理(例如給控件賦值),因?yàn)镈oWork事件跟正常的界面的事件不同,這個(gè)事件在非UI線程上執(zhí)行,所以才能異步執(zhí)行。

Step 3. 生成ProgressChanged事件并添加控件處理的代碼,因?yàn)檫@個(gè)事件是在UI線程上執(zhí)行的,所以可以給界面中的控件進(jìn)行賦值等操作。

Step 4. 如果需要,請生成RunWorkerCompleted事件,在此處理異步執(zhí)行結(jié)束的業(yè)務(wù)邏輯。當(dāng)然,此事件也是在UI線程上執(zhí)行的,可以給界面中的控件進(jìn)行賦值等操作。

Step 5. 在需要執(zhí)行異步操作的地方調(diào)用RunWorkerAsync方法,開始執(zhí)行異步調(diào)用。

下面是具體的代碼:

  1. public Form1()     
  2.  {     
  3.    InitializeComponent();    
  4.      bWorker.DoWork += new DoWorkEventHandler(bWorker_DoWork);    
  5.      bWorker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bWorker_RunWorkerCompleted);   
  6.      bWorker.ProgressChanged += new ProgressChangedEventHandler(bWorker_ProgressChanged);     
  7.    this.Text = "UI thread id is:" + Thread.CurrentThread.ManagedThreadId.ToString();     
  8.  }     
  9. BackgroundWorker bWorker = new BackgroundWorker();    
  10. void bWorker_DoWork(object sender, DoWorkEventArgs e)    
  11.  {    
  12.  int tick = (int)e.Argument;    
  13.   Thread thr = Thread.CurrentThread;    
  14.  for (int i = 0; i < 30; i++)    
  15.   {    
  16.     if (bWorker.CancellationPending)    
  17.      {    
  18.         e.Cancel = true;    
  19.           //break;    
  20.    }    
  21.  else    
  22.     {    
  23.            Thread.Sleep(TimeSpan.FromSeconds(tick));    
  24.           bWorker.ReportProgress(i, DateTime.Now.ToString() + "\\TID:" + thr.ManagedThreadId.ToString());    
  25.       }    
  26.     }    
  27.        
  28.  }   
  29. void bWorker_ProgressChanged(object sender, ProgressChangedEventArgs e)    
  30.  {    
  31. progressBar1.Value = e.ProgressPercentage;  
  32.      label1.Text = e.UserState.ToString();   
  33.  }    
  34. void bWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)  
  35.  {    
  36.  label1.Text = DateTime.Now.ToString();   
  37.      progressBar1.Value = progressBar1.Maximum;    
  38.      if (e.Cancelled)    
  39.  label1.Text = "User cancelled.";    
  40. }    
  41. private void btnInvoke_Click(object sender, EventArgs e)   
  42. {   
  43.   bWorker.WorkerReportsProgress = true;   
  44.     bWorker.WorkerSupportsCancellation = true;    
  45.     if (!bWorker.IsBusy)   
  46.      bWorker.RunWorkerAsync(1);    
  47. }   
  48. private void btnCancel_Click(object sender, EventArgs e)    
  49.  {    
  50.     if (bWorker.WorkerSupportsCancellation)   
  51.         bWorker.CancelAsync();    

上面的代碼請注意幾個(gè)地方:

1. 第50行,開始調(diào)用RunWorkerAsync方法前,請先判斷IsBusy屬性是否是false,因?yàn)槿绻麨閠rue,則說明上一次的調(diào)用還沒有結(jié)束,再次調(diào)用會(huì)引發(fā)異常。

2. 第56行,調(diào)用CancelAsync方法前,請先設(shè)置WorkerSupportsCancellation屬性為true,否則會(huì)引發(fā)異常。

3. 第26行,調(diào)用ReportProgress方法前,請先設(shè)置WorkerReportsProgress屬性為true,否則會(huì)引發(fā)異常。

4. RunWorkerAsync方法傳遞的參數(shù)是object類型,這個(gè)參數(shù)的值可以在DoWork事件的參數(shù)e中的屬性Argument獲得。

5. ReportProgress方法傳遞的參數(shù)可以在事件ProgressChanged中的參數(shù)e中獲得。

6. 調(diào)用CancelAsync方法只是向后臺(tái)的異步線程發(fā)出結(jié)束申請,具體什么時(shí)候結(jié)束,由線程自動(dòng)管理。

7. 在RunWorkerCompleted事件中,如果想知道后臺(tái)任務(wù)是正常執(zhí)行完畢還是被調(diào)用CancelAsync方法強(qiáng)制中斷,請參考事件的參數(shù)e的Cancelled屬性。(奇怪的是這個(gè)屬性不會(huì)在你調(diào)用CancelAsync方法后自動(dòng)設(shè)置為true,你需要象代碼中的20行那樣進(jìn)行設(shè)置。)

8. 請注意第7行和第26行的代碼,這兩段代碼中的線程的ID,說明了DoWork事件和UI是在兩個(gè)不同的線程上執(zhí)行。

實(shí)際上BackgroundWorker并非直接用來解決跨線程的控件調(diào)用的問題,只是它提供了一種工作機(jī)制,可以讓你的程序利用它來執(zhí)行異步調(diào)用,并且在異步調(diào)用的過程中進(jìn)行控件的操作。

好了,關(guān)于如何對界面中的控件進(jìn)行跨線程的調(diào)用就介紹這么多吧,希望對大家有所幫助。

原文標(biāo)題: 在.Net中進(jìn)行跨線程的控件操作(下篇:BackgroundWorker)

鏈接:http://www.cnblogs.com/happinessCodes/archive/2010/07/22/1783199.html

【編輯推薦】

  1. .NET上執(zhí)行多線程應(yīng)該注意的兩點(diǎn)
  2. C#多線程控制進(jìn)度條之多線程安全
  3. 通過多線程為基于.NET的應(yīng)用程序?qū)崿F(xiàn)響應(yīng)迅速
  4. .NET 4.0中任務(wù)與線程關(guān)系談
  5. 在.NET多線程程序中使用異步調(diào)用的簡易方法
責(zé)任編輯:彭凡 來源: 博客園
相關(guān)推薦

2024-05-16 12:51:15

WinForms線程UI

2010-01-04 14:49:30

Silverlight

2009-08-27 13:38:36

C#線程相關(guān)問題

2009-07-24 10:36:08

ASP.NET控件

2025-02-10 07:05:00

WinFormUI線程

2009-07-17 17:33:22

jQuery

2009-12-30 17:29:53

Silverlight

2024-05-27 00:27:59

WinForm線程應(yīng)用程序

2009-12-21 17:35:24

ADO.NET對象

2010-01-06 17:02:28

.Net Framew

2009-08-04 13:39:43

ASP.NET 2.0

2010-01-21 10:12:57

VB.NET控件自動(dòng)排

2010-03-16 19:29:26

Java多線程操作

2009-08-27 13:55:08

C#子線程

2009-12-22 10:15:17

ADO.NET規(guī)則

2009-12-30 11:13:28

ADO.NET操作

2009-12-28 15:46:22

ADO.NET操作

2010-01-06 15:43:22

.Net Framew

2009-08-07 17:42:43

ASP.NET Dat

2011-08-30 13:40:28

MySQL線程
點(diǎn)贊
收藏

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