C#使用BackgroundWorker控件

在我們的程序中,經常會有一些耗時較長的運算,為瞭保證用戶體驗,不引起界面不響應,我們一般會采用多線程操作,讓耗時操作在後臺完成,完成後再進行處理或給出提示,在運行中,也會時時去刷新界面上的進度條等顯示,必要時還要控制後臺線程中斷當前操作。

在.net中,提供瞭一個組件BackgroundWorker就是專門解決這個問題的。BackgroundWorker類允許在單獨的專用線程上運行操作。 耗時的操作(如下載和數據庫事務)在長時間運行時可能會導致用戶界面(UI)似乎處於停止響應狀態。如果需要能進行響應的用戶界面,而且面臨與這類操作相關的長時間延遲,則可以使用BackgroundWorker類方便地解決問題。

程序執行步驟:

  • 1、調用BackgroundWorker的RunWorkerAsync()方法,如果後臺操作需要參數,在調用RunWorkerAsync()方法時給出參數,在DoWork事件處理程序內部,可以從DoWorkEventArgs.Argument屬性中提取該參數。
  • 2、執行DoWork事件,後臺需要執行的代碼放到DoWork事件裡面執行。當調用RunWorkerAsync()方法時,BackgroundWorker通過觸發DoWork事件,開始執行後臺操作

顯示後臺操作進度:

為瞭顯示後臺操作的執行進度,首先要使WorkerReportsProgress等於true,然後調用BackgroundWorker的ReportProgress()方法,通過它傳遞操作完成的進度值,此外,該方法觸發ProgressChanged事件,在此事件中,通過ProgressChangedEventArgs的實例,接收到主線程傳遞過來的參數。

取消後臺操作:

為瞭使 BackgroundWorker 可以取消後臺正在執行的操作,首先要把屬性WorkerSupportsCancellation 的值設置為 true。接著調用CancelAsync()方法,該方法使得屬性CancellationPending 為true,利用CancellationPending 屬性,可以判斷是否取消後臺異步操作。

後臺操作完成後,反饋給用戶:

當後臺操作完成以後,無論是completed 還是cancelled,RunWorkerCompleted()事件都會被觸發,通過此方法可以將後臺操作的完成結果反饋給用戶。RunWorkerCompleted 事件處理函數會在DoWork 事件處理函數返回後被調用。通過它我們可以進行一些運算結束後的操作,比如禁用取消按鈕,異常處理,結果顯示等。註意,如果想要拿到e.Result,您需要在BGWorker_DoWork方法中設置 e.Result屬性另外,通過RunWorkerCompletedEventArgs實例的Cancelled 屬性,以判斷是否是Cancel操作使得後臺操作終止;

從後臺操作返回值

在執行DoWork事件時DoWorkEventArgs實例的Result屬性,返回值到用戶;在RunWorkerCompleted事件裡,RunWorkerCompletedEventArgs 實例的Result屬性接收值;

創建BackgroundWorkerDemo例子:

  • 1.新建一個windows窗體應用程序,如:BackgroundWorkerDemo
  • 2.拖一個ProgressBar(進度條)和一個BackgroundWorker控件到Form窗體上,界面如圖:

後臺代碼:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.IO;
using System.Threading;

namespace BackgroundWorkerDemo
{
    public partial class FrmDemo : Form
    {
        //設置生成臨時文件的路徑
        static string strSaveDir = @"F:\培訓";
        public FrmDemo()
        {
            InitializeComponent();

            //顯示後臺操作的執行進度
            this.bgWork.WorkerReportsProgress = true;
            //可以取消後臺正在執行的操作
            this.bgWork.WorkerSupportsCancellation = true;
        }

        /// <summary>
        /// 開始
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btn_Start_Click(object sender, EventArgs e)
        {
            if (Directory.Exists(strSaveDir) == false)
            {
                return;
            }
            btn_Start.Enabled = false;
            int count = Convert.ToInt32(this.txt_File.Text.ToString().Trim());
            //設置進度條
            this.proBar.Minimum = 0;
            this.proBar.Maximum = count;
            this.proBar.Value = this.proBar.Minimum;
            //開始執行異步線程,進行後臺操作,給後臺傳遞參數
            this.bgWork.RunWorkerAsync(count);
        }

        /// <summary>
        /// 後臺操作要處理的任務代碼
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void bgWork_DoWork(object sender, DoWorkEventArgs e)
        {
            //獲取從RunWorkerAsync()方法裡面傳遞的參數的值
            int fileCount= Convert.ToInt32(e.Argument);
            Random rand = new Random();
            byte[] buffer = new byte[2048];
            for (int i = 0; i < fileCount; i++)
            {
                try
                {
                    string strFileName = Path.Combine(strSaveDir, i.ToString() + ".tmp");
                    using (var stream = File.Create(strFileName))
                    {
                        int n = 0;
                        int maxByte = 8 * 1024 * 1024;
                        while (n < maxByte)
                        {
                            rand.NextBytes(buffer);
                            stream.Write(buffer, 0, buffer.Length);
                            n += buffer.Length;
                        }
                    }
                }
                catch (Exception ex)
                {
                    continue;
                }
                finally
                {
                    //報告進度
                    this.bgWork.ReportProgress(i + 1);
                    Thread.Sleep(100);
                }

                //判斷是否取消瞭後臺操作
                if (bgWork.CancellationPending)
                {
                    e.Cancel = true;
                    return;
                }

                //設置返回值
                e.Result = 234;
            }
        }

        /// <summary>
        /// 更新前臺界面進度條
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void bgWork_ProgressChanged(object sender, ProgressChangedEventArgs e)
        {
            //獲取異步任務的進度百分百
            int val = e.ProgressPercentage;
            this.label2.Text = string.Format("已經生成{0}個文件", val);
            //進度條顯示當前進度
            this.proBar.Value = val;
        }

        /// <summary>
        /// 後臺操作完成,向前臺反饋信息
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void bgWork_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
        {
            btn_Start.Enabled = true;
            //用戶取消操作(e.Cancelled==true,表示異步操作已被取消)
            if (e.Cancelled)
            {
                MessageBox.Show("用戶取消後臺操作", "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);
            }
            else
            {
                MessageBox.Show("操作完成", "提示", MessageBoxButtons.OK, MessageBoxIcon.Information);

                //接收返回值
                int result = (int)e.Result;

                MessageBox.Show("返回值:" + result);
            }
        }

        /// <summary>
        /// 取消
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void btn_Cancle_Click(object sender, EventArgs e)
        {
            //調用CancelAsync(),取消掛起的後臺操作
            this.bgWork.CancelAsync();
        }
    }
}

運行界面:

操作完成界面:

接收返回值:

取消後臺操作:

 到此這篇關於C#使用BackgroundWorker控件的文章就介紹到這瞭。希望對大傢的學習有所幫助,也希望大傢多多支持WalkonNet。

推薦閱讀: