WinForm中異步加載數據並使用進度條


在WinForm程序中,有時會因為加載大量數據導致UI界面假死,這種情況對於用戶來說是非常不友好的。因此,在加載大量數據的情況下,首先應該將數據加載放在另一線程中進行,這樣保證了UI界面的響應;其次可以提供一個進度條使用戶明白程序正在加載數據,同時清楚知道目前加載的進度。

實現上述功能的一個簡單的方式是利用 System.ComponentModel 中的工具類:BackgroundWorker,它支持取消,進度報告,異常轉發,並且實現了 IComponent 接口,意味着可以直接在VS設計器中從工具箱中拖到界面上使用。

下面以一個例子來說明如何使用 BackgroundWorker,更詳細的 BackgroundWorker 說明可以參考Threading in C#(或者 中文翻譯):

1. UI界面添加一個進度條,一個開始按鈕,一個結束按鈕,以及BackgroundWorker,並設置下列 BackgroundWorker 屬性(例子中設置了其Name為bw):

 

  • WorkerReportsProcess:默認為False,將其設置為True,支持進度報告
  • WorkerSupportsCancellation:默認為False,將其設置為True,支持取消

2. DoWork事件,在其中執行我們的數據加載,我們執行一個循環來模擬數據加載

private void bw_DoWork(object sender, DoWorkEventArgs e)
{
    var count = (int)e.Argument;
    for (int i = 1; i <= count; i++)
    {
        if (bw.CancellationPending)
        {
            e.Cancel = true;
            return;
        }
                    
        bw.ReportProgress(i);
        Thread.Sleep(200);      // 模擬耗時的任務
    }                          
}
  • 注意:在這個方法中不能進行UI控件的更新。
  • 通過檢查 CancellationPending 來判斷用戶是否進行了取消
  • 通過調用 ReportProgress 來報告進度

 

3. ProgressChanged 事件,在這里可以操作進度條,以及其他UI控件。

private void bw_ProgressChanged(object sender, ProgressChangedEventArgs e)
{
    progressBar.Value = e.ProgressPercentage;
    resultTextBox.Text += DateTime.Now + "\r\n";
}

通過e.ProgressPercentage來獲取任務執行過程中設置的進度,以此來更新進度條。

4. RunWorkerCompleted 事件,在這里可以更新UI,以及進行異常處理。

private void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
{
    if (e.Cancelled)
        resultTextBox.Text += "任務取消。" + "\r\n";
    else if (e.Error != null)
        resultTextBox.Text += "出現異常: " + e.Error + "\r\n";
    else
        resultTextBox.Text += "任務完成。 " + "\r\n";
}

當執行過程中出現異常時,異常會被轉發到這里,因此可以在這里處理異常。

5. 通過一個開始按鈕調和一個取消按鈕來控制:

  • bw.RunWorkerAsync()   啟動
  • bw.CancelAsync()  取消

下邊是完整的代碼及輸出:

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();
        bw.DoWork += bw_DoWork;
        bw.ProgressChanged += bw_ProgressChanged;
        bw.RunWorkerCompleted += bw_RunWorkerCompleted;
    }

    private void startButton_Click(object sender, EventArgs e)
    {
        progressBar.Value = 0;
        progressBar.Maximum = 10;

        resultTextBox.Text = "任務開始..." + "\r\n";
        bw.RunWorkerAsync(10);
    }

    private void bw_DoWork(object sender, DoWorkEventArgs e)
    {
        var count = (int)e.Argument;
        for (int i = 1; i <= count; i++)
        {
            if (bw.CancellationPending)
            {
                e.Cancel = true;
                return;
            }

            if (i == 2)
                throw new Exception("出錯啦!");
                    
            bw.ReportProgress(i);
            Thread.Sleep(200);      
        }                          
    }

    private void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
    {
        if (e.Cancelled)
            resultTextBox.Text += "任務取消。" + "\r\n";
        else if (e.Error != null)
            resultTextBox.Text += "出現異常: " + e.Error + "\r\n";
        else
            resultTextBox.Text += "任務完成。 " + "\r\n";
    }

    private void bw_ProgressChanged(object sender, ProgressChangedEventArgs e)
    {
        progressBar.Value = e.ProgressPercentage;
        resultTextBox.Text += DateTime.Now + "\r\n";
    }

    private void cancelbutton_Click(object sender, EventArgs e)
    {
        bw.CancelAsync();
    }
}

輸出如下:

 

參考:Threading in C# --> 中文翻譯


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM