凡是WinForm的应用程序,如果他执行了一个的非常冗长的处理操作(比如文件查询),它在执行时会锁定用户界面,虽然主活动窗口 一直在运行,但用户无法与程序交互,无法移动窗体或改变窗体大小,所以用户感觉很不爽。如何做才能使得这个程序有响应。答案就是在后台线程中执行这个操 作。
在这里已经有了多种方法来做这个事情:
(一)委托异步调用
将具体耗时的操作作为一个委托,并用BeginInvoke来异步执行这个委托(Invoke是同步调用),并且可以为这个操作传入参数并且通过EndInvoke方法获得返回返回值。
(二)使用ThreadPool
新建.net FrameWork中自带的WaitCallback委托,然后放到线程池中运行ThreadPool.QueueUserWorkItem( callback ); 根据WaitCallback委托的定义,可以传入一个object类型的参数。
但是不能精确的控制线程池中的线程。
(三)使用Thread
和ThreadPool相比,使用Thread的开销会比较大。但是它有它的优势,使用 Thread 类可以显式管理线程。只要有可能,就应该使用 ThreadPool 类来创建线程。然而,在一些情况下,您还是需要创建并管理您自己的线程,而不是使用 ThreadPool 类。在.net 2.0 中,提供了一个新的委托 ParameterizedThreadStart 支持启动一个线程并传入参数,这是对原来的ThreadStart委托的改进。
说了这么多还没有说到今天的主角BackgroundWorker,他也是一个在2.0中新增的类,可以用于启动后台线程,并在后台计算结束后调用主线程 的方法.可以看出同样的功能使用委托的异步调用也可以实现,只是使用BackgroundWorker的话会更加的简便快捷,可以节省开发时间,并把你从 创建自己的委托以及对它们的调用中解救出来。真是这样的吗看看下面这个例子。其实我也是从101Samples中看到的例子。
先看看BackgroundWorker中的主要概念。
第一:主要的事件及参数。
DoWork——当执行BackgroundWorker.RunWorkerAsync方法时会触发该事件,并且传递DoWorkEventArgs参数;
ProgressChanged——操作处理中获得的处理状态变化,通过BackgroundWorker.ReportProgress(int)方法 触发该事件,并且传递ProgressChangedEventArgs,其中包含了处理的百分比;
RunWorkerCompleted ——异步操作完成后会触发该事件,当然如果需要在操作过程中结束可以执行BackgroundWorker.CancelAsync方法要求异步调用中 止,并且在异步委托操作中检测BackgroundWorker.CancellationPending属性如果为true的话,跳出异步调用,同时将 DoWorkEventArgs.Cancel属性设为true,这样当退出异步调用的时候,可以让处理RunWorkerCompleted事件的函数 知道是正常退出还是中途退出。
第二:主要的方法。
BackgroundWorker.RunWorkerAsync——
“起动”异步调用的方法有两次重载RunWorkerAsync(),RunWorkerAsync(object argument),第二个重载提供了一个参数,可以供异步调用使用。(如果有多个参数要传递怎么办,使用一个类来传递他们吧)。调用该方法后会触发 DoWork事件,并且为处理DoWork事件的函数DoWorkEventArg事件参数,其中包含了RunWorkerAsync传递的参数。在相应 DoWork的处理函数中就可以做具体的复杂操作。
BackgroundWorker.ReportProgress——
有时候需要在一个冗长的操作中向用户不断反馈进度,这样的话就可以调用的ReportProgress(int percent),在调用 ReportProgress 方法时,触发ProgressChanged事件。提供一个在 0 到 100 之间的整数,它表示后台活动已完成的百分比。你也可能提供任何对象作为第二个参数,允许你 给事件处理程序传递状态信息。作为传递到此过程的 ProgressChangedEventArgs 参数属性,百分比和你自己的对象(如果提供的话)均要被传递到 ProgressChanged 事件处理程序。这些属性被分别命名为 ProgressPercentage 和 UserState,并且你的事件处理程序可以以任何需要的方式使用它们。(注意:只有在 BackgroundWorker.WorkerReportsProgress属性被设置为true该方法才可用)。
BackgroundWorker.CancelAsync——
但需要退出异步调用的时候,就调用的这个方法。但是样还不够,因为它仅仅是将BackgroudWorker.CancellationPending属 性设置为true。你需要在具体的异步调用处理的时候,不断检查BackgroudWorker.CancellationPending是否为 true,如果是真的话就退出。(注意:只有在BackgroundWorker.WorkerSupportsCancellation属性被设置为 true该方法才可用)。
贴出一段101Samples里面的代码,看一下就明白了:
public partial class MainForm : Form
{
private System.ComponentModel.BackgroundWorker backgroundCalculator;
public MainForm()
{
InitializeComponent();
backgroundCalculator = new BackgroundWorker();
backgroundCalculator.WorkerReportsProgress = true;
backgroundCalculator.WorkerSupportsCancellation = true;
backgroundCalculator.DoWork +=
new DoWorkEventHandler(backgroundCalculator_DoWork);
backgroundCalculator.ProgressChanged +=
new ProgressChangedEventHandler(backgroundCalculator_ProgressChanged);
backgroundCalculator.RunWorkerCompleted +=
new RunWorkerCompletedEventHandler(backgroundCalculator_RunWorkerCompleted);
updateStatus(String.Empty);
}
private int getNextPrimeAsync(int start, BackgroundWorker worker, DoWorkEventArgs e)
{
int percentComplete = 0;
start++;
while (!isPrime(start))
{
// Check for cancellation
if (worker.CancellationPending)
{
e.Cancel = true;
break;
}
else
{
start++;
percentComplete++;
worker.ReportProgress(percentComplete % 100);
}
}
return start;
}
void backgroundCalculator_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
{
if (e.Cancelled)
{
updateStatus("Cancelled.");
}
else if (e.Error != null)
{
reportError(e.Error);
}
else
{
reportPrime((int)e.Result);
}
calcProgress.Value = 0;
}
void backgroundCalculator_ProgressChanged(object sender, ProgressChangedEventArgs e)
{
updateProgress(e.ProgressPercentage);
}
void backgroundCalculator_DoWork(object sender, DoWorkEventArgs e)
{
int start = (int) e.Argument;
e.Result = getNextPrimeAsync(start, (BackgroundWorker)sender, e);
}
private void nextPrimeAsyncButton_Click(object sender, EventArgs e)
{
updateStatus("Calculating...");
int start;
Int32.TryParse(textBoxPrime.Text, out start);
if (start == 0)
{
reportError("The number must be a valid integer");
}
else
{
// Kick off the background worker process
backgroundCalculator.RunWorkerAsync(int.Parse(textBoxPrime.Text));
}
}
private void cancelButton_Click(object sender, EventArgs e)
{
if (backgroundCalculator.IsBusy)
{
updateStatus("Cancelling...");
backgroundCalculator.CancelAsync();
}
}
// Update the Status label
private void updateStatus(string status)
{
calcStatus.Text = status;
}
// Indicate progress using progress bar
private void updateProgress(int percentComplete)
{
calcProgress.Value = percentComplete;
}
}
BackgroundWorker创建自己的委托并调用这个窗体的 Invoke 方法来运行它,BackgroundWorker 组件以一种优雅的方式来处理这个线程转换。BackgroundWorker 组件允许你从后台线程中调用它的 ReportProgress 方法,该方法触发其 ProgressChanged 事件处理例程返回到窗体的线程中。你不必使用 delegate/Invoke 方法自己处理这个线程转换,而是调用 ReportProgress,其余的事情交给组件来做。
分享到:
相关推荐
2. **UI更新挑战**:由于跨线程访问UI控件是不安全的,你需要使用Control.Invoke或Control.BeginInvoke来确保在UI线程上执行更新操作,这增加了代码的复杂性。 3. **错误处理**:异常处理需要手动实现,通常需要捕获...
`BackgroundWorker`类是.NET Framework的System.ComponentModel命名空间的一部分,它允许在后台线程上执行操作,而不会阻塞UI线程。 `BackgroundWorker`的主要特点和功能包括: 1. **异步执行任务**:`...
在.NET框架中,BackgroundWorker组件是一个非常实用的工具,它允许在不同的线程上执行耗时的操作,同时保持用户界面(UI)的响应性。在标题为"关于backgroundworker的使用方法"的描述中,提到使用BackgroundWorker来...
在VB.NET编程中,`BackgroundWorker`组件是一个非常重要的工具,它允许我们在应用程序的主线程之外执行耗时操作,防止用户界面(UI)线程被阻塞,从而保持UI的响应性。`BackgroundWorker`是.NET框架提供的一种异步...
在.NET框架中,BackgroundWorker组件是一个非常实用的工具,它使得在用户界面(UI)线程上执行耗时操作变得更加简单。本篇文章将详细介绍如何使用BackgroundWorker,以及它在多线程编程中的重要性。 首先,理解背景...
6. **线程安全**:由于 `BackgroundWorker` 在后台线程上运行,所有对 UI 控件的操作都必须在主线程中进行。使用 `Control.Invoke` 或 `Control.BeginInvoke` 方法将后台线程的结果安全地传递到 UI 线程。 在实际...
这些操作如果在主线程上执行,会阻塞UI(用户界面),导致程序响应迟钝,用户体验下降。为了解决这个问题,.NET Framework引入了`BackgroundWorker`组件,它使得在后台线程执行耗时任务成为可能,同时保持UI的响应性...
在VB.NET编程中,`BackgroundWorker`组件是一个非常重要的工具,它允许我们在应用程序的后台线程上执行耗时操作,而不会阻塞用户界面(UI)。这样,即使在执行大量计算或I/O操作时,用户仍能与应用程序交互。本教程...
BackgroundWorker组件是为了简化多线程编程而设计的,它提供了事件驱动的模型,可以在后台线程上执行耗时操作,并在完成后更新UI。在DoWork事件中进行后台处理,在ProgressChanged或RunWorkerCompleted事件中更新UI...
【C#的BackgroundWorker控件】是.NET框架中用于实现多线程操作的重要组件,尤其在用户界面(UI)更新和后台任务执行方面起到关键作用。它的主要目标是在不阻塞用户界面的情况下运行耗时的任务,提升应用程序的响应性...
在Android和iOS平台上,虽然没有直接对应的BackgroundWorker类,但它们提供了类似的概念和技术来实现后台处理。让我们深入探讨如何在Android和iOS中实现后台任务。 ### Android中的后台处理 在Android中,我们可以...
在.NET框架中,`BackgroundWorker`组件是一种便捷的机制,用于在UI线程(主线程)之外执行耗时的操作,从而避免阻塞用户界面。它主要用于实现简单的异步操作,尤其是在Windows Forms或WPF应用程序中。`...
BackgroundWorker类允许在单独的专用线程上运行操作。 耗时的操作(如下载和数据库事务)在长时间运行时可能会导致用户界面(UI)似乎处于停止响应状态。如果需要能进行响应的用户界面,而且面临与这类操作相关的长...
BackgroundWorker 使用异步编程模型,允许在单独的线程上执行任务,而不影响UI线程。这意味着在执行长时间运行的任务时,用户仍可以与应用程序交互,如滚动、点击按钮等,避免了程序假死现象。 二、主要属性 1. `Is...
`BackgroundWorker`组件是.NET Framework提供的一种简单而有效的方式,用于在单独的线程上执行耗时操作,同时允许用户界面(UI)保持响应。下面将详细阐述`BackgroundWorker`的使用,包括如何控制其开始、暂停、继续...
`BackgroundWorker`是.NET Framework提供的一种多线程解决方案,允许在后台线程上执行耗时操作,而不会冻结用户界面。它通过事件模型来报告工作进度和完成情况,非常适合用于初始化进度条这样的场景。 1. **...
在C#编程中,`BackGroundWorker`组件是.NET Framework提供的一种用于实现多线程操作的工具,尤其适合于在用户界面(UI)线程上执行长时间运行的任务,避免阻塞UI,提升用户体验。本小例子展示了如何巧妙地运用`...
BackgroundWorker组件是.NET框架中一个非常实用的工具,主要用于在UI线程之外执行耗时操作,从而避免阻塞用户界面。这个组件使得开发者能够轻松地实现异步编程,确保应用程序的响应性和用户体验。下面我们将详细探讨...
在Windows Forms或WPF等桌面应用开发中,`BackgroundWorker`组件是一个常用工具,它允许开发者在后台线程上执行耗时操作,同时保持用户界面的响应性。本主题将深入探讨如何利用`BackgroundWorker`实现异步处理,并在...