溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

winform多線程組件BackgroundWorker怎么使用

發(fā)布時間:2022-05-11 13:39:12 來源:億速云 閱讀:237 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要介紹了winform多線程組件BackgroundWorker怎么使用的相關(guān)知識,內(nèi)容詳細(xì)易懂,操作簡單快捷,具有一定借鑒價值,相信大家閱讀完這篇winform多線程組件BackgroundWorker怎么使用文章都會有所收獲,下面我們一起來看看吧。

BackgroundWorker是·net里用來執(zhí)行多線程任務(wù)的控件,它允許編程者在一個單獨的線程上執(zhí)行一些操作。

可以通過編程方式創(chuàng)建 BackgroundWorker,也可以將它從“工具箱”的“組件”選項卡中拖到窗體上。 如果在 Windows 窗體設(shè)計器中創(chuàng)建 BackgroundWorker,則它會出現(xiàn)在組件欄中,而且它的屬性會顯示在“屬性”窗口中。

常用方法

  • RunWorkerAsync 開始執(zhí)行后臺操作。引發(fā) DoWork 事件。
       public void RunWorkerAsync(); //啟動線程,觸發(fā)DoWork事件
       public void RunWorkerAsync(object argument);

  • CancelAsync 請求取消掛起的后臺操作。
    注意:這個方法是將 CancellationPending 屬性設(shè)置為 true,并不會終止后臺操作。在后臺操作中要檢查CancellationPending 屬性,來決定是否要繼續(xù)執(zhí)行耗時的操作。

  • ReportProgress引發(fā) ProgressChanged 事件。
       public void ReportProgress(int percentProgress); //報告進(jìn)度,觸發(fā)ProgressChanged事件
       public void ReportProgress(int percentProgress, object userState);

常用屬性

  • IsBusy:  //只讀屬性,用來判斷當(dāng)前線程是否正在工作中。

  • CancellationPending: 指示應(yīng)用程序是否已請求取消后臺操作。只讀屬性,默認(rèn)為 false,當(dāng)執(zhí)行了 CancelAsync 方法后,值為 true。

  • WorkerSupportsCancellation:指示是否支持異步取消。要執(zhí)行 CancelAsync 方法,需要先設(shè)置該屬性為 true。

  • WorkerReportsProgress:指示是否能報告進(jìn)度。要執(zhí)行 ReportProgress 方法,需要先設(shè)置該屬性為 true。

常用事件

  • DoWork: 調(diào)用 RunWorkerAsync 方法時發(fā)生。

  • ProgressChanged:可選,調(diào)用 ReportProgress 方法時發(fā)生。

  • RunWorkerCompleted:可選,后臺操作已完成、被取消或引發(fā)異常時發(fā)生。

注意:在 DoWork 事件處理程序中不操作任何用戶界面對象。而應(yīng)該通過 ProgressChanged 和RunWorkerCompleted 事件與用戶界面進(jìn)行通信。

如果想在 DoWork 事件處理程序中和用戶界面的控件通信,可在用 ReportProgress 方法。ReportProgress(int percentProgress, object userState),可以傳遞一個對象。ProgressChanged 事件可以從參數(shù)ProgressChangedEventArgs 類的UserState 屬性得到這個信息對象。這個事件也可以實現(xiàn)進(jìn)度條功能,把任務(wù)的進(jìn)度實時呈現(xiàn)給用戶。

簡單的程序用BackgroundWorker 比 Thread 方便,Thread中和用戶界面上的控件通信比較麻煩,需要用委托來調(diào)用控件的 Invoke 或BeginInvoke 方法,沒有 BackgroundWorker 方便。

使用backgroundWorker步驟

  • 新建BackgroundWorder對象;

  • 根據(jù)需求, 設(shè)置是否能取消(WorkerSupportsCancellation)、是否報告進(jìn)度(WorkerReportsProgress);

  • 根據(jù)需求,設(shè)置好相關(guān)事件,DoWorker、ProgressChanged、ProgressChanged;

  • 調(diào)用RunWorkerAsyns()方法,啟動線程;

  • 在需要取消的位置,判斷CancellationPending的值,并做相關(guān)處理;//可選

  • 在適當(dāng)?shù)奈恢谜{(diào)用ReportProgress(int percentProgress)方法,報告進(jìn)度。

BackgroundWorker實例

    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();

            backgroundWorker1.WorkerReportsProgress = true;//報告完成進(jìn)度
            backgroundWorker1.WorkerSupportsCancellation = true;//允許用戶終止后臺線程
                                                                //綁定事件
            backgroundWorker1.DoWork += new DoWorkEventHandler(backgroundWorker1_DoWork);
            backgroundWorker1.ProgressChanged += new ProgressChangedEventHandler(backgroundWorker1_ProgressChanged);
            backgroundWorker1.RunWorkerCompleted += new RunWorkerCompletedEventHandler(backgroundWorker1_RunWorkerCompleted);

        }
        //開始按鈕
        private void button1_Click(object sender, EventArgs e)
        {
            if (!backgroundWorker1.IsBusy)//判斷backgroundWorker1是否正在運行異步操作
            {
                backgroundWorker1.RunWorkerAsync(1000);//開始執(zhí)行后臺異步操作,調(diào)用DoWork事件
            }
            while (backgroundWorker1.IsBusy)//等待后臺運行完畢
            {
                Application.DoEvents();
            }
            MessageBox.Show("操作完成");
        }

        //取消按鈕
        private void button2_Click(object sender, EventArgs e)
        {
            if (backgroundWorker1.WorkerSupportsCancellation == true)
            {
                backgroundWorker1.CancelAsync();//取消后臺操作
                backgroundWorker1.Dispose();//釋放資源
            }
        }

        //DoWork事件聲明要執(zhí)行的耗時操作
        private void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e)
        {
            BackgroundWorker backgroundWorker = sender as BackgroundWorker;
            e.Result = ListNumber(backgroundWorker, e);//運算結(jié)果保存在e.Result中(在RunWorkerCompleted事件可能會使用到)
        }

        bool ListNumber(object sender, DoWorkEventArgs e)
        {
            int num = (int)e.Argument;//接收傳入的參數(shù),即RunWorkerAsync(object argument)傳入的值
            for (int i = 1; i <= num; i++)
            {
                if (backgroundWorker1.CancellationPending)//判斷是否請求了取消后臺操作,如果為false則退出
                {
                    e.Cancel = true;//e.Cancel 是否應(yīng)該取消事件
                    return false;
                }
                Thread.Sleep(10);//執(zhí)行一個耗時操作
                backgroundWorker1.ReportProgress(i * 100 / num, i);//報告完成進(jìn)度

            }
            return true;
        }

        private void backgroundWorker1_ProgressChanged(object sender, ProgressChangedEventArgs e)
        {

            progressBar1.Value = e.ProgressPercentage;//將完成進(jìn)度數(shù)據(jù)傳給進(jìn)度條
            label1.Text = e.ProgressPercentage + "%";
            //將中間計算結(jié)果在ListBox控件中顯示出來
            listBox1.Items.Add(e.UserState);//接收ReportProgress方法傳遞過來的userState
        }

        private void backgroundWorker1_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
        {
            if (e.Cancelled == true) //e.Cancelled指示異步操作是否已被取消
            {
                MessageBox.Show("Canceled!");
            }
            else if (e.Error != null) //e.Error 指示異步操作期間發(fā)生的錯誤
            {
                MessageBox.Show("Error: " + e.Error.Message);
            }
            else
            {
                MessageBox.Show(e.Result.ToString());  //e.Result 獲取異步操作結(jié)果的值,即DoWork事件中,Result設(shè)置的值。
            }

        }
    }

對Winform窗體控件進(jìn)行安全調(diào)用

(WinForm中實現(xiàn)多線程的首選方法是backgroundWorker)

Thread thread = new Thread(SetLabel);//另開一個線程上設(shè)置Label的屬性
thread.Start();
 
delegate void Action(string args);//Net3.5自帶的Action無參數(shù)無返回值
private void SetLabel()
{
    Action action = delegate (string args)
    {
        this.label1.Text = args;
    };
    if (this.InvokeRequired)//判斷當(dāng)前代碼是運行于創(chuàng)建該控件的線程之上,還是運行于另一個線程之上。
        this.Invoke(action, "XXX");//注意控件的Inovoke和BeginInvoke委托的方法都執(zhí)行在UI線程上。對Control的Invoke方法可以使用一個簡單的系統(tǒng)委托MethodInvoker無參數(shù)返回Void調(diào)用。
    else
        action("XXX");
}

關(guān)于“winform多線程組件BackgroundWorker怎么使用”這篇文章的內(nèi)容就介紹到這里,感謝各位的閱讀!相信大家對“winform多線程組件BackgroundWorker怎么使用”知識都有一定的了解,大家如果還想學(xué)習(xí)更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI