博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
C#调用耗时函数时显示进度条浅探
阅读量:5011 次
发布时间:2019-06-12

本文共 5240 字,大约阅读时间需要 17 分钟。

最近在做一个VSS日志分析工具,使用C#进行开发,在完成了所有功能后,发现,从服务器下载VSS日志非常耗时,因为此,导致工具使用体验不好,所以,准备增加一个进度条。
鉴于C#不经常使用,一下子搞个进度条貌似比较难,而且其他的开发任务也在一并进行,所以,昨天一天,并没有多大的进展。
今天,是周末,正好可以利用,在查阅了大量网上资料以及实例后,我制作了几个实例,以备后来之用。
使用C#显示进度条,涉及到多线程编程,我只探索了使用BackgroundWorker和Thread的方法,下面分别列出。

第一种:使用BackgroundWorker进行进度条控制
BackgroundWorker对象有三个主要的事件:
DoWork - 当BackgroundWorker对象的多线程操作被执行时触发。
RunWokerCompleted - 当BackgroundWoker对象的多线程操作完成时触发。
ProgressChanged - 当BackgroundWorker对象的多线程操作状态改变时触发。
WorkerReportsProgress - 如果想让BackgroundWorker对象以异步的方式报告线程实时进度,必须将该属性的值设为true。
BackgroundWorker对象的ReportProgress方法用于向主线程返回后台线程执行的实时进度。

实例代码一,控制主窗体中的进度条显示

public partial class Form1 : Form

{
/// <summary>
/// 后台线程
/// </summary>
private BackgroundWorker bkWorker = new BackgroundWorker();

/// <summary>

/// 步进值
/// </summary>
private int percentValue = 0;

public Form1()

{
InitializeComponent();

bkWorker.WorkerReportsProgress = true;

bkWorker.WorkerSupportsCancellation = true;
bkWorker.DoWork += new DoWorkEventHandler(DoWork);
bkWorker.ProgressChanged += new ProgressChangedEventHandler(ProgessChanged);
bkWorker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(CompleteWork);
}

private void btnStart_Click(object sender, EventArgs e)

{
percentValue = 10;
this.progressBar1.Maximum = 1000;
// 执行后台操作
bkWorker.RunWorkerAsync();
}

public void DoWork(object sender, DoWorkEventArgs e)

{
// 事件处理,指定处理函数
e.Result = ProcessProgress(bkWorker, e);
}

public void ProgessChanged(object sender, ProgressChangedEventArgs e)

{
// bkWorker.ReportProgress 会调用到这里,此处可以进行自定义报告方式
this.progressBar1.Value = e.ProgressPercentage;
int percent = (int)(e.ProgressPercentage / percentValue);
this.label1.Text = "处理进度:" + Convert.ToString(percent) + "%";
}

public void CompleteWork(object sender, RunWorkerCompletedEventArgs e)

{
this.label1.Text = "处理完毕!";
}

private int ProcessProgress(object sender, DoWorkEventArgs e)

{
for (int i = 0; i <= 1000; i++)
{
if (bkWorker.CancellationPending)
{
e.Cancel = true;
return -1;
}
else
{
// 状态报告
bkWorker.ReportProgress(i);

// 等待,用于UI刷新界面,很重要

System.Threading.Thread.Sleep(1);
}
}

return -1;

}
}

 

实例代码二,控制弹出窗体中的进度条显示

主窗体代码:

public partial class Form1 : Form

{
private BackgroundWorker bkWorker = new BackgroundWorker();
private Form2 notifyForm = new Form2();

public Form1()

{
InitializeComponent();

// 使用BackgroundWorker时不能在工作线程中访问UI线程部分,

// 即你不能在BackgroundWorker的事件和方法中操作UI,否则会抛跨线程操作无效的异常
// 添加下列语句可以避免异常。
CheckForIllegalCrossThreadCalls = false;

bkWorker.WorkerReportsProgress = true;

bkWorker.WorkerSupportsCancellation = true;
bkWorker.DoWork += new DoWorkEventHandler(DoWork);
bkWorker.ProgressChanged += new ProgressChangedEventHandler(ProgessChanged);
bkWorker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(CompleteWork);
}

private void btnStart_Click(object sender, EventArgs e)

{
notifyForm.StartPosition = FormStartPosition.CenterParent;

bkWorker.RunWorkerAsync();

notifyForm.ShowDialog();
}

public void DoWork(object sender, DoWorkEventArgs e)

{
// 事件处理,指定处理函数
e.Result = ProcessProgress(bkWorker, e);
}

public void ProgessChanged(object sender, ProgressChangedEventArgs e)

{
// bkWorker.ReportProgress 会调用到这里,此处可以进行自定义报告方式
notifyForm.SetNotifyInfo(e.ProgressPercentage, "处理进度:" + Convert.ToString(e.ProgressPercentage) + "%");
}

public void CompleteWork(object sender, RunWorkerCompletedEventArgs e)

{
notifyForm.Close();

MessageBox.Show("处理完毕!");

}

private int ProcessProgress(object sender, DoWorkEventArgs e)

{
for (int i = 0; i <= 1000; i++)
{
if (bkWorker.CancellationPending)
{
e.Cancel = true;
return -1;
}
else
{
// 状态报告
bkWorker.ReportProgress(i / 10);

// 等待,用于UI刷新界面,很重要

System.Threading.Thread.Sleep(1);
}
}

return -1;

}
}

 

子窗体代码:

public partial class Form2 : Form

{
public Form2()
{
InitializeComponent();
}

public void SetNotifyInfo(int percent, string message)

{
this.label1.Text = message;
this.progressBar1.Value = percent;
}
}

 

 

第二种,使用Thread来实现

使用Thread实现,虽然步骤上比较麻烦,但是调用流程比较简单,也是一种可以参考的方法
使用时,首先要定义代理以及函数,然后实现线程函数,在线程函数中调用代理,最后启动线程,传入线程函数。
下面是实例代码:

 

public partial class Form1 : Form

{
private Form2 progressForm = new Form2();
// 代理定义,可以在Invoke时传入相应的参数
private delegate void funHandle(int nValue);
private funHandle myHandle = null;

public Form1()

{
InitializeComponent();
}

private void btnStart_Click(object sender, EventArgs e)

{
// 启动线程
System.Threading.Thread thread = new System.Threading.Thread(new System.Threading.ThreadStart(ThreadFun));
thread.Start();
}

/// <summary>

/// 线程函数中调用的函数
/// </summary>
private void ShowProgressBar()
{
myHandle = new funHandle(progressForm.SetProgressValue);
progressForm.ShowDialog();
}

/// <summary>

/// 线程函数,用于处理调用
/// </summary>
private void ThreadFun()
{
MethodInvoker mi = new MethodInvoker(ShowProgressBar);
this.BeginInvoke(mi);

System.Threading.Thread.Sleep(1000); // sleep to show window

for (int i = 0; i < 1000; ++i)

{
System.Threading.Thread.Sleep(5);
// 这里直接调用代理
this.Invoke(this.myHandle, new object[] { (i / 10) });
}
}
}

 

子窗体代码

public partial class Form2 : Form

{
public Form2()
{
InitializeComponent();
}

public void SetProgressValue(int value)

{
this.progressBar1.Value = value;
this.label1.Text = "Progress :" + value.ToString() + "%";

// 这里关闭,比较好,呵呵!

if (value == this.progressBar1.Maximum - 1) this.Close();
}
}

转载于:https://www.cnblogs.com/ZLGBloge/p/4208837.html

你可能感兴趣的文章
校外实习报告(九)
查看>>
android之android.intent.category.DEFAULT的用途和使用
查看>>
CAGradientLayer 透明渐变注意地方(原创)
查看>>
【OpenSource】--Web Bench 1.5
查看>>
python-1:工欲善其事,必先利其器 安装配置Anaconda x32位 更新后,启动不了,解决方案(亲测)...
查看>>
Linux 命令(二) Linux下查看文件文件内容命令
查看>>
ubuntu系统如何屏幕截图
查看>>
ArcGIS Engine 创建索引(属性索引)——提高查询效率
查看>>
栅格数据AE
查看>>
开发笔记
查看>>
织梦DEDE多选项筛选_联动筛选功能的实现_二次开发
查看>>
SQL Server 排名函数 简单应用
查看>>
Could not find file "/var/www/default/bin\roslyn\csc.exe".
查看>>
关于软件工程的疑问
查看>>
iOS关于RunLoop和Timer
查看>>
SQL处理层次型数据的策略对比:Adjacency list vs. nested sets: MySQL【转载】
查看>>
已存在同名的数据库,或指定的文件无法打开或位于 UNC 共享目录中。
查看>>
MySQL的随机数函数rand()的使用技巧
查看>>
thymeleaf+bootstrap,onclick传参实现模态框中遇到的错误
查看>>
python字符串实战
查看>>