学习C#中的BackgroundWorker 组件

1. BackgroundWorker 组件概述

许多经常执行的操作可能需要很长的执行时间。 例如:

  • 图像下载

  • Web 服务调用

  • 文件下载和上载(包括点对点应用程序)

  • 复杂的本地计算

  • 数据库事务

  • 本地磁盘访问(相对于内存访问来说其速度很慢)

此类操作可能会导致用户界面在运行时进行阻止。 如果你需要能进行响应的 UI,而且面临与这类操作相关的长时间延迟,BackgroundWorker 组件可以提供一种方便的解决方案。

使用 BackgroundWorker 组件,你可以在不同于应用程序的主 UI 线程的另一线程上异步("在后台")执行耗时的操作。 若要使用 BackgroundWorker,只需要告诉该组件要在后台执行的耗时的辅助方法,然后调用 RunWorkerAsync 方法。 在辅助方法以异步方式运行的同时,你的调用线程将继续正常运行。 该方法运行完毕后,BackgroundWorker 通过引发 RunWorkerCompleted 事件(可选择包含操作结果)可向调用线程发出警报。

BackgroundWorker 组件可通过"工具箱"的"组件"选项卡获得。要将 BackgroundWorker 添加到窗体,请将 BackgroundWorker 组件拖到你的窗体上。 该组件出现在组件栏中,而其属性将显示在"属性"窗口中。

若要启动异步操作,请使用 RunWorkerAsync 方法。 RunWorkerAsync 采用一个可选 object 参数,该参数可用于将变量传递给辅助方法。 BackgroundWorker 类公开 DoWork 事件,你的辅助线程通过 DoWork 事件处理程序附加到该事件。

DoWork 事件处理程序采用一个 DoWorkEventArgs 参数,该参数具有 Argument 属性。 此属性接收来自 RunWorkerAsync 的参数,并可以传递给 DoWork 事件处理程序中调用的辅助方法。 以下示例显示了如何分配名为 ComputeFibonacci 的辅助方法的结果。 它是一个更大示例的一部分,可以在如何:实现使用后台操作的窗体中找到该示例。

cs 复制代码
// This event handler is where the actual,
// potentially time-consuming work is done.
private void backgroundWorker1_DoWork(object sender,
    DoWorkEventArgs e)
{
    // Get the BackgroundWorker that raised this event.
    BackgroundWorker worker = sender as BackgroundWorker;

    // Assign the result of the computation
    // to the Result property of the DoWorkEventArgs
    // object. This is will be available to the
    // RunWorkerCompleted eventhandler.
    e.Result = ComputeFibonacci((int)e.Argument, worker, e);
}

2.示例

cs 复制代码
using System;
using System.ComponentModel;
using System.Windows.Forms;

namespace BackgroundWorkerExample
{
    public partial class Form1 : Form
    {
        private BackgroundWorker backgroundWorker = new BackgroundWorker();

        public Form1()
        {
            InitializeComponent();

            // 配置 BackgroundWorker
            backgroundWorker.WorkerReportsProgress = false; // 如果不需要报告进度,设置为 false
            backgroundWorker.WorkerSupportsCancellation = false; // 如果不需要取消操作,设置为 false
            backgroundWorker.DoWork += new DoWorkEventHandler(backgroundWorker_DoWork);
            backgroundWorker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(backgroundWorker_RunWorkerCompleted);
        }

        private void backgroundWorker_DoWork(object sender, DoWorkEventArgs e)
        {
            // 在这里执行耗时的操作,例如计算斐波那契数列
            int n = (int)e.Argument; // 获取从 RunWorkerAsync 传递的参数
            int result = ComputeFibonacci(n);
            e.Result = result; // 将结果存储在 Result 属性中
        }

        private int ComputeFibonacci(int n)
        {
            if (n <= 1)
            {
                return n;
            }
            return ComputeFibonacci(n - 1) + ComputeFibonacci(n - 2);
        }

        private void backgroundWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
        {
            // 操作完成后,在这里更新 UI
            if (e.Error != null)
            {
                MessageBox.Show("操作过程中出现错误: " + e.Error.Message);
            }
            else if (e.Cancelled)
            {
                MessageBox.Show("操作已取消。");
            }
            else
            {
                MessageBox.Show("斐波那契数列的第 " + ((int)backgroundWorker.Argument) + " 项是: " + e.Result);
            }
        }

        private void buttonStart_Click(object sender, EventArgs e)
        {
            int number;
            if (int.TryParse(textBoxNumber.Text, out number))
            {
                backgroundWorker.Argument = number; // 设置传递给 DoWork 的参数
                buttonStart.Enabled = false; // 禁用按钮以避免重复点击
                backgroundWorker.RunWorkerAsync(); // 启动后台操作
            }
            else
            {
                MessageBox.Show("请输入一个有效的整数。");
            }
        }
    }
}
相关推荐
huangdong_16 小时前
淘宝商品SKU图自动分类技术深度解析:从DOM解析到智能归档
开发语言·javascript·ecmascript
阿正的梦工坊16 小时前
【Rust】12-借用检查器与非词法生命周期
开发语言·后端·rust
qq_25183645716 小时前
基于java Web网络订餐系统设计与实现 源码文档
java·开发语言·前端
xmtxz16 小时前
计算机网络基础课程学习心得:从理论抽象到硬核实战的进阶之路
运维·学习
秋916 小时前
3年经验Python后端转AI Engineer:3个月实战转型计划(2026版)
开发语言·人工智能·python
凡人叶枫16 小时前
Effective C++ 条款17:以独立语句将 newed 对象置入智能指针
java·linux·开发语言·c++·算法
飞天狗11117 小时前
零基础JavaWeb入门——第2课:让网页“活”起来 —— JSP是什么?
java·开发语言·前端·后端·web
YM52e17 小时前
男孩子在外自我保护指南——用鸿蒙 ArkTS 构建交互式安全教育应用
学习·安全·华为·harmonyos·鸿蒙·鸿蒙系统
醇氧17 小时前
【Linux】Java 服务生产级部署指南:实现常驻后台、开机自启与系统服务化管理
java·开发语言
凡人叶枫18 小时前
Effective C++ 条款16:成对使用 new 和 delete 时要采取相同形式
开发语言·c++·effective c++