在文本框中輸入一個(gè)數(shù)字,點(diǎn)擊開(kāi)始累加按鈕,程序計(jì)算從1開(kāi)始累計(jì)到該數(shù)字的結(jié)果。因?yàn)樵摾奂舆^(guò)程比較耗時(shí),如果直接在UI線程中進(jìn)行,那么當(dāng)前窗口將出現(xiàn)假死。為了有更好的用戶體驗(yàn),程序啟動(dòng)一個(gè)新的線程來(lái)單獨(dú)執(zhí)行該計(jì)算,然后每隔200毫秒讀取一次累加結(jié)果,并把結(jié)果顯示到文本框下方的label控件中。同時(shí),程序支持取消操作,點(diǎn)擊取消累計(jì)按鈕,程序?qū)⑷∠奂硬僮鳎旬?dāng)前累加值顯示到label中。為了方便后面的描述,我把UI線程稱作主線程,把執(zhí)行累加計(jì)算的線程稱作工作者線程。該過(guò)程有兩個(gè)關(guān)鍵點(diǎn):
1:如何在工作者線程中訪問(wèn)主線程創(chuàng)建的控件;
2:如何取消比較耗時(shí)的計(jì)算;
為了便于在工作者線程中調(diào)用累加過(guò)程,我把它寫(xiě)成一個(gè)單獨(dú)方法,如下:
1:方法需要傳遞一個(gè)CancellationToken參數(shù),用于支持取消操作(《clr via c# 3版》中把這種方式稱作協(xié)作式取消,也就是說(shuō)某一個(gè)操作必須支持取消,然后才能取消該操作);
2:為了允許工作者線程訪問(wèn)主線程創(chuàng)建的lbl_Status控件,我在該線程中使用this.Invoke方法。該方法用于獲得主線程所創(chuàng)建控件的訪問(wèn)權(quán)。它需要一個(gè)委托作為參數(shù),在該委托中我們可以定義對(duì)lbl_Status的操作。例如在上例中我就是把當(dāng)前的累加結(jié)果賦給lbl_Status的Text屬性。
然后我們看一下如何在一個(gè)共走著線程中執(zhí)行計(jì)算耗時(shí)的操作,也就是“開(kāi)始累加”按鈕的操作:
我使用線程池線程來(lái)執(zhí)行該操作,之所以使用線程池線程而不是自己的Threading對(duì)象,是因?yàn)榫€程池默認(rèn)已經(jīng)為我們創(chuàng)建好了一些線程,從而省去創(chuàng)建新線程造成的一些列資源消耗,同時(shí),完成計(jì)算任務(wù)后該線程池線程自動(dòng)回到池中等待下一個(gè)任務(wù)。我把_cts作為一個(gè)成員變量,聲明如下:
它需要引入using System.Threading;命名空間。
取消操作更加簡(jiǎn)單,代碼如下:
這樣我們就完成了在winform中使用多線程的例子,同時(shí)該例子支持取消操作。完整代碼如下:
namespace WinformApp
{
public partial class Form1 : Form
{
private CancellationTokenSource _cts;
public Form1()
{
InitializeComponent();
}
/// <summary>
/// 從1累加到指定的值,為了讓該方法支持取消操作所以需要CancellationToken參數(shù)
/// </summary>
/// <param name="countTo">累加到的指定值</param>
/// <param name="ct">取消憑證</param>
private void CountTo(int countTo, CancellationToken ct) {
int sum = 0;
for (; countTo > 0; countTo--) {
if (ct.IsCancellationRequested) {
break;
}
sum += countTo;
//Invoke方法用于獲得創(chuàng)建lbl_Status的線程所在的上下文
this.Invoke(new Action(()=>lbl_Status.Text = sum.ToString()));
Thread.Sleep(200);
}
}
private void btn_Count_Click(object sender, EventArgs e)
{
_cts = new CancellationTokenSource();
ThreadPool.QueueUserWorkItem(state=>CountTo(int.Parse(txt_CountTo.Text),_cts.Token));
}
private void btn_Cancel_Click(object sender, EventArgs e)
{
if (_cts != null)
_cts.Cancel();
}
private void btn_Pause_Click(object sender, EventArgs e)
{
}
}
}
新聞熱點(diǎn)
疑難解答
圖片精選