WPF之BackgroundWorker
BackgroundWorker类允许您在单独的线程上执行某个可能导致用户界面(UI)停止响应的耗时操作,下面来介绍一下这个线程类BackgroundWorker,大家可以结合这位大佬的这篇文章,说的比较仔细
https://www.cnblogs.com/zhaoshujie/p/9634136.html
1.属性
//bool类型,指示应用程序是否已请求取消后台操作。此属性通常放在用户执行的异步操作内部,用来判断用户是否取消执行异步操作。当执行BackgroundWorker.CancelAsync()方法时,该属性值将变为True。
public bool CancellationPending { get; }
//bool类型,指示BackgroundWorker是否正在执行一个异步操作。此属性通常放在BackgroundWorker.RunWorkerAsync()方法之前,避免多次调用RunWorkerAsync()方法引发异常。当执行BackgroundWorker.RunWorkerAsync()方法是,该属性值将变为True。public bool IsBusy { get; }
//bool类型,指示BackgroundWorker是否可以报告进度更新。当该属性值为True是,将可以成功调用ReportProgress方法,默认为false
public bool WorkerReportsProgress { get; set; }
//bool类型,指示BackgroundWorker是否支持异步取消操作。当该属性值为True是,将可以成功调用CancelAsync方法取消工作线程。
public bool WorkerSupportsCancellation { get; set; }
2.方法
//取消线程操作,属性CancellationPending将变为True
public void CancelAsync ();
//执行该方法,将触发ProgressChanged事件
public void ReportProgress (int percentProgress);
//开始执行线程操作,触发DoWork事件,执行事件中的代码
public void RunWorkerAsync ();
3.事件
//当执行RunWorkerAsync方法的时候触发该事件
public event System.ComponentModel.DoWorkEventHandler DoWork;
//当执行ReportProgress方法的时候触发该事件
public event System.ComponentModel.ProgressChangedEventHandler ProgressChanged;
//当线程操作完成、或者被取消、或者抛出异常的时候触发该事件
public event System.ComponentModel.RunWorkerCompletedEventHandler RunWorkerCompleted;
4.接下来通过一个开始/结束/暂停/继续执行线程的例子来说明
namespace LoginUI
{
/// <summary>
/// MainWindow.xaml 的交互逻辑
/// </summary>
public partial class LoadingWindow : Window
{private BackgroundWorker _backgroundWorker;
//初始化为终止状态(true),那么该方法将一直工作
private ManualResetEvent manualReset = new ManualResetEvent(true);public LoadingWindow()
{
InitializeComponent();
}
private void Window_Loaded(object sender, RoutedEventArgs e)
{
#region
InitializeBackgroundWorker();
btnSatrt.Content = "开始";
btnStop.Content = "暂停";
btnEnd.Content = "结束";
#endregion
} //执行后台线程
private void btnSatrt_Click(object sender, RoutedEventArgs e)
{
//开始执行后台操作
_backgroundWorker.RunWorkerAsync();
btnSatrt.IsEnabled = false;
btnEnd.IsEnabled = true;
} //暂停/继续后台线程
private void btnStop_Click(object sender, RoutedEventArgs e)
{
if (btnStop.Content.ToString() == "暂停")
{
manualReset.Reset();//暂停当前线程的工作,发信号给waitOne方法,阻塞
btnStop.Content = "继续";
}
else
{
manualReset.Set();//继续当前线程的工作,发信号给waitOne方法,继续运行Waitone后面的代码
btnStop.Content = "暂停";
}
}
//结束后台线程
private void btnEnd_Click(object sender, RoutedEventArgs e)
{
_backgroundWorker.CancelAsync();
btnSatrt.IsEnabled = true;
btnEnd.IsEnabled = false;
}
private void InitializeBackgroundWorker()
{
#region
if (_backgroundWorker == null)
_backgroundWorker = new BackgroundWorker();
//bool类型,指示BackgroundWorker是否可以报告进度更新。当该属性值为True时,将可以成功调用ReportProgress方法
_backgroundWorker.WorkerReportsProgress = true;
//bool类型,指示BackgroundWorker是否支持异步取消操作。当该属性值为True是,将可以成功调用CancelAsync方法
_backgroundWorker.WorkerSupportsCancellation = true;
//执行RunWorkerAsync方法后触发DoWork,将异步执行backgroundWorker_DoWork方法中的代码
_backgroundWorker.DoWork += new DoWorkEventHandler(backgroundWorker_DoWork);
//执行ReportProgress方法后触发ProgressChanged,将执行ProgressChanged方法中的代码
_backgroundWorker.ProgressChanged += new ProgressChangedEventHandler(_backgroundWorker_ProgressChanged);
//异步操作完成或取消时执行的操作,当调用DoWork事件执行完成时触发。
_backgroundWorker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(backgroundWorker_RunWorkerCompleted);
#endregion
} //backgroundWorker_DoWork完成后执行
private void backgroundWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
{
if((bool)e.Result==true)
{
progressBar.Value = 100;
lstbLog.Items.Add("操作已完成");
MessageBox.Show("操作已完成");
}
} private void _backgroundWorker_ProgressChanged(object sender, ProgressChangedEventArgs e)
{
//ProgressPercentage是ReportProgress方法中第一个参数传递过来的
progressBar.Value = e.ProgressPercentage;
//lstbLog是一个listView控件,UserState是ReportProgress方法中第二个参数传递过来的
lstbLog.Items.Add(e.UserState.ToString());
} //backgroundWorker_DoWork是运行在非UI线程上的,因此该内部代码应该避免与UI界面交互,交互的逻辑应该放到
//ProgressChanged和RunWorkerCompleted事件中
private void backgroundWorker_DoWork(object sender, DoWorkEventArgs e)
{
for (int i=0;i<=100;i++)
{
////如果ManualResetEvent的初始化为终止状态(true),那么该方法将一直工作,
//直到收到Reset信号,将阻塞。然后,直到收到Set信号,就继续工作。
manualReset.WaitOne();
//可以通过调用CancelAsync方法设置CancellationPending的值为false
if (_backgroundWorker.CancellationPending)
{
_backgroundWorker.ReportProgress(i, String.Format("{0}%,操作被用户申请结束", i));
_backgroundWorker.ReportProgress(0, String.Format("已完成{0}%",0 ));
e.Result = false;
break;
}
//调用 ReportProgress 方法,会触发ProgressChanged事件
_backgroundWorker.ReportProgress(i, String.Format("已完成{0}%", i));
System.Threading.Thread.Sleep(50);
if (i == 100)
{
//执行DoWork方法完成,设置为True
e.Result = true;
}
}
}
点击开始按钮,开始执行
点击结束,将会结束上述任务

点击暂停/继续按钮可以暂停/继续当前任务

上述每个按钮的功能逻辑还没有完善,但是基本的功能都实现了。本人也是个新手,如有什么问题麻烦指出,谢谢各位大佬!
WPF之BackgroundWorker的更多相关文章
- WPF 分页控件 WPF 多线程 BackgroundWorker
WPF 分页控件 WPF 多线程 BackgroundWorker 大家好,好久没有发表一篇像样的博客了,最近的开发实在头疼,很多东西无从下口,需求没完没了,更要命的是公司的开发从来不走正规流程啊, ...
- WPF线程(Step2)——BackgroundWorker
在WPF中第二个常用的线程处理方式就是BackgroundWorker. 以下是BackgroundWorker一个简单的例子. public partial class MainWindow : W ...
- 让你的WPF程序使用多线程——BackgroundWorker
在wpf中可以使用许多方法执行异步操作.利用.NET的芳芳就是手动创建一个新的System.Threading.Thread对象,提供一步代码,并使用THread.Start()方法加载代码.这种方法 ...
- 【WPF/C#】使用BackgroundWorker实现多线程/异步操作
做WPF时需要做一个异步加载时的Loading遮罩,搜Stackoverflow看到很多方法,看到了这个插件: BusyIndicator in the extended WPF Toolkit 同时 ...
- WPF 精修篇 BackgroundWorker
原文:WPF 精修篇 BackgroundWorker 效果 <Grid> <Grid.RowDefinitions> <RowDefinition Height=&qu ...
- WPF 耗时操作时,加载loging 动画 (BackgroundWorker 使用方法)
1.定义一个全局 BackgroundWorker private System.ComponentModel.BackgroundWorker bgMeet0; 2.设置执行耗时的任务为True b ...
- WPF 后台任务 等待动画 样例 && C# BackgroundWorker 详解
运行效果: 前台代码: <Window x :Class="Waiting.Window1" xmlns="http://schemas.microsoft.com ...
- Wpf Backgroundworker
<Window x:Class="WpfApp53.MainWindow" xmlns="http://schemas.microsoft.com/winfx/20 ...
- WPF 线程 Dispatcher
WPF 应用程序从两个线程开始: 一个用于处理呈现 一个用于管理 UI 呈现线程有效地隐藏在后台运行,而UI线程则接收输入.处理事件.绘制屏幕以及运行应用程序代码. 大多数应用程序都使用一个 UI 线 ...
随机推荐
- 字节一面:说说TCP的三次握手
上周有朋友去了字节面试,问到了TCP三次握手的问题,当时回答的不是很好,对于三次握手的发送的报文信息都不太熟,本文主要做一下总结和记录. TCP全称为Transmission Control Prot ...
- Spring(四)-声明式事务
Spring-04 声明式事务 1.事务的定义 事务就是由一组逻辑上紧密关联的多个工作单元(数据库操作)而合并成一个整体,这些操作要么都执行,要么都不执行. 2.事务的特性:ACID 1)原子性A : ...
- Javaweb__Jquery
今日内容 1. JQuery 基础: 1. 概念 2. 快速入门 3. JQuery对象和JS对象区别与转换 4. 选择器 5. DOM操作 6. 案例 JQuery 基础: 1. 概念: 一个Jav ...
- InnoDB_锁总结
1. 查询会对资源添加共享锁 加了共享锁的资源不可以被修改:但可以被查询(也是会在资源上再加共享锁) 2. 数据修改会对资源添加排他锁 加了排他锁的资源只能被持有这个排他锁的事务读取和修改,其他事务读 ...
- Latex中也能展示动态图?
技术背景 在学术领域,很多文档是用Latex做的,甚至有很多人用Latex Beamer来做PPT演示文稿.虽然在易用性和美观等角度来说,Latex Beamer很大程度上不如PowerPoint,但 ...
- 使用Prometheus和Grafana监控nacos集群
官方文档:https://nacos.io/zh-cn/docs/monitor-guide.html 按照部署文档搭建好Nacos集群 配置application.properties文件,暴露me ...
- 在Portainer上管理其他docker主机(这只是其中一种方式),另一种方式看这个文档:使用Portainer管理其他主机的docker应用有两种方式
其他主机开启远程连接docker端口 需要设置一下2375端口的监听.通过修改docker配置文件方式进行监听. 修改配置文件修改监听端口 使用Centos7安装的docker,所以下面的配置是适用于 ...
- 用VS Code搞Qt 6:Gui基础类型——QGuiApplication和QWindow
在99.996%的情况下,我们弄 Qt 应用都会使用 QApplication 类和 QWidget 类,即直接用 Widgets 库中的组件/控件.为了方便开发人员自己造轮子,Qt 也提供了一套基础 ...
- 密码学奇妙之旅、03 HMAC单向散列消息认证码、Golang代码
HMAC 单向散列消息认证码 消息认证码MAC是用于确认完整性并进行认证的技术,消息认证码的输入包括任意长度的消息和一个发送者和接收者之间共享的密钥(可能还需要共享盐值). HMAC是使用单向散列函数 ...
- C++自学笔记 构造与析构;
构造与析构 类不是实体:对象属于类:函数属于类 : 用不同的对象调用同一个类里面的函数的时候,函数知道是哪一个对象在调用它 关键字 this this是一个指针 Point a; a.print(); ...
