WinForm:进度条的实现(异步)

在WinForm中经常遇到一些费时的操作界面,比如统计某个磁盘分区的文件夹或者文件数目,如果分区很大或者文件过多的话,处理不好就会造成“假死”的情况,或者报“线程间操作无效”的异常,为了解决这个问题,可以使用委托来处理,在.net2.0中还可以用BackgroundWorker类。

BackgroundWorker类是.net 2.0里新增加的一个类,对于需要长时间操作而不需要用户长时间等待的情况可以使用这个类。

注意确保在 DoWork 事件处理程序中不操作任何用户界面对象。而应该过 ProgressChanged 和 RunWorkerCompleted 事件与用户界面进行通信。

 它有几个属性:
CancellationPending——指示应用程序是否已请求取消后台操作。
IsBusy——指示 BackgroundWorker 是否正在运行异步操作
WorkerReportsProgress——该值指示 BackgroundWorker 能否报告进度更新
WorkerSupportsCancellation——该值指示 BackgroundWorker 是否支持异步取消

还有如下事件:
DoWork——调用 RunWorkerAsync 时发生。
ProgressChanged——调用 ReportProgress 时发生。
RunWorkerCompleted——当后台操作已完成、被取消或引发异常时发生。

还有如下方法:
CancelAsync——请求取消挂起的后台操作
ReportProgress——引发 ProgressChanged 事件
RunWorkerAsync——开始执行后台操作

关于BackgroundWorker类更多详细介绍可参考:http://blog.csdn.net/zhzuo/archive/2008/07/23/2699305.aspx

下载与本文相关的ProcessTest示例相关代码。
https://files.cnblogs.com/Jason_z/ProcessTest.rar
程序运行界面


主要代码如下:
 

复制代码
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;

namespace ProcessTest
{
    
public partial class Form1 : Form
    {
        
public Form1()
        {
            InitializeComponent();

            worker.WorkerReportsProgress 
= true;

            worker.WorkerSupportsCancellation 
= true;

            
//正式做事情的地方
            worker.DoWork+=new DoWorkEventHandler(DoWork);

            
//任务完称时要做的,比如提示等等
            worker.ProgressChanged += new ProgressChangedEventHandler(ProgessChanged);

            
//任务进行时,报告进度
            worker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(CompleteWork);
        }

        
//调用 RunWorkerAsync 时发生
        public void DoWork(object sender, DoWorkEventArgs e)
        {
            e.Result 
= ComputeFibonacci(worker, e);
            
//获取异步操作结果的值,当ComputeFibonacci(worker, e)返回时,异步过程结束
        }

        
//调用 ReportProgress 时发生
        public void ProgessChanged(object sender, ProgressChangedEventArgs e)
        {
            
this.progressBar1.Value = e.ProgressPercentage;
            
//将异步任务进度的百分比赋给进度条
        }

        
//当后台操作已完成、被取消或引发异常时发生
        public void CompleteWork(object sender, RunWorkerCompletedEventArgs e)
        {
            MessageBox.Show(
"完成!");
        }

        
private int ComputeFibonacci(object sender, DoWorkEventArgs e)
        {
            
for (int i = 0; i < 1000; i++)
            {
                
//判断应用程序是否取消后台操作
                if(worker.CancellationPending)
                {
                    e.Cancel 
= true;
                }
                
else
                {
                    worker.ReportProgress(i);
                }

                System.Threading.Thread.Sleep(
10);
            }
            
return -1;
        }

        
private void btnStart_Click(object sender, EventArgs e)
        {
            worker.RunWorkerAsync();

            btnStart.Enabled 
= false;

            btnPause.Enabled 
= true;
        }

        
private void btnPause_Click(object sender, EventArgs e)
        {
            btnPause.Enabled 
= false;

            btnStart.Enabled 
= true;

            worker.CancelAsync();
        }
    }
}
复制代码

 

posted @   Jason.z  阅读(6888)  评论(2编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 25岁的心里话
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示