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

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

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

  1. public partial class Form1 : Form
  2. {
  3. /// <summary>
  4. /// 后台线程
  5. /// </summary>
  6. private BackgroundWorker bkWorker = new BackgroundWorker();
  7. /// <summary>
  8. /// 步进值
  9. /// </summary>
  10. private int percentValue = 0;
  11. public Form1()
  12. {
  13. InitializeComponent();
  14. bkWorker.WorkerReportsProgress = true;
  15. bkWorker.WorkerSupportsCancellation = true;
  16. bkWorker.DoWork += new DoWorkEventHandler(DoWork);
  17. bkWorker.ProgressChanged += new ProgressChangedEventHandler(ProgessChanged);
  18. bkWorker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(CompleteWork);
  19. }
  20. private void btnStart_Click(object sender, EventArgs e)
  21. {
  22. percentValue = 10;
  23. this.progressBar1.Maximum = 1000;
  24. // 执行后台操作
  25. bkWorker.RunWorkerAsync();
  26. }
  27. public void DoWork(object sender, DoWorkEventArgs e)
  28. {
  29. // 事件处理,指定处理函数
  30. e.Result = ProcessProgress(bkWorker, e);
  31. }
  32. public void ProgessChanged(object sender, ProgressChangedEventArgs e)
  33. {
  34. // bkWorker.ReportProgress 会调用到这里,此处可以进行自定义报告方式
  35. this.progressBar1.Value = e.ProgressPercentage;
  36. int percent = (int)(e.ProgressPercentage / percentValue);
  37. this.label1.Text = "处理进度:" + Convert.ToString(percent) + "%";
  38. }
  39. public void CompleteWork(object sender, RunWorkerCompletedEventArgs e)
  40. {
  41. this.label1.Text = "处理完毕!";
  42. }
  43. private int ProcessProgress(object sender, DoWorkEventArgs e)
  44. {
  45. for (int i = 0; i <= 1000; i++)
  46. {
  47. if (bkWorker.CancellationPending)
  48. {
  49. e.Cancel = true;
  50. return -1;
  51. }
  52. else
  53. {
  54. // 状态报告
  55. bkWorker.ReportProgress(i);
  56. // 等待,用于UI刷新界面,很重要
  57. System.Threading.Thread.Sleep(1);
  58. }
  59. }
  60. return -1;
  61. }
  62. }

下面是运行结果

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

  1. public partial class Form1 : Form
  2. {
  3. private BackgroundWorker bkWorker = new BackgroundWorker();
  4. private Form2 notifyForm = new Form2();
  5. public Form1()
  6. {
  7. InitializeComponent();
  8. // 使用BackgroundWorker时不能在工作线程中访问UI线程部分,
  9. // 即你不能在BackgroundWorker的事件和方法中操作UI,否则会抛跨线程操作无效的异常
  10. // 添加下列语句可以避免异常。
  11. CheckForIllegalCrossThreadCalls = false;
  12. bkWorker.WorkerReportsProgress = true;
  13. bkWorker.WorkerSupportsCancellation = true;
  14. bkWorker.DoWork += new DoWorkEventHandler(DoWork);
  15. bkWorker.ProgressChanged += new ProgressChangedEventHandler(ProgessChanged);
  16. bkWorker.RunWorkerCompleted += new RunWorkerCompletedEventHandler(CompleteWork);
  17. }
  18. private void btnStart_Click(object sender, EventArgs e)
  19. {
  20. notifyForm.StartPosition = FormStartPosition.CenterParent;
  21. bkWorker.RunWorkerAsync();
  22. notifyForm.ShowDialog();
  23. }
  24. public void DoWork(object sender, DoWorkEventArgs e)
  25. {
  26. // 事件处理,指定处理函数
  27. e.Result = ProcessProgress(bkWorker, e);
  28. }
  29. public void ProgessChanged(object sender, ProgressChangedEventArgs e)
  30. {
  31. // bkWorker.ReportProgress 会调用到这里,此处可以进行自定义报告方式
  32. notifyForm.SetNotifyInfo(e.ProgressPercentage, "处理进度:" + Convert.ToString(e.ProgressPercentage) + "%");
  33. }
  34. public void CompleteWork(object sender, RunWorkerCompletedEventArgs e)
  35. {
  36. notifyForm.Close();
  37. MessageBox.Show("处理完毕!");
  38. }
  39. private int ProcessProgress(object sender, DoWorkEventArgs e)
  40. {
  41. for (int i = 0; i <= 1000; i++)
  42. {
  43. if (bkWorker.CancellationPending)
  44. {
  45. e.Cancel = true;
  46. return -1;
  47. }
  48. else
  49. {
  50. // 状态报告
  51. bkWorker.ReportProgress(i / 10);
  52. // 等待,用于UI刷新界面,很重要
  53. System.Threading.Thread.Sleep(1);
  54. }
  55. }
  56. return -1;
  57. }
  58. }

子窗体代码

  1. public partial class Form2 : Form
  2. {
  3. public Form2()
  4. {
  5. InitializeComponent();
  6. }
  7. public void SetNotifyInfo(int percent, string message)
  8. {
  9. this.label1.Text = message;
  10. this.progressBar1.Value = percent;
  11. }
  12. }

下面是运行结果


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

  1. public partial class Form1 : Form
  2. {
  3. private Form2 progressForm = new Form2();
  4. // 代理定义,可以在Invoke时传入相应的参数
  5. private delegate void funHandle(int nValue);
  6. private funHandle myHandle = null;
  7. public Form1()
  8. {
  9. InitializeComponent();
  10. }
  11. private void btnStart_Click(object sender, EventArgs e)
  12. {
  13. // 启动线程
  14. System.Threading.Thread thread = new System.Threading.Thread(new System.Threading.ThreadStart(ThreadFun));
  15. thread.Start();
  16. }
  17. /// <summary>
  18. /// 线程函数中调用的函数
  19. /// </summary>
  20. private void ShowProgressBar()
  21. {
  22. myHandle = new funHandle(progressForm.SetProgressValue);
  23. progressForm.ShowDialog();
  24. }
  25. /// <summary>
  26. /// 线程函数,用于处理调用
  27. /// </summary>
  28. private void ThreadFun()
  29. {
  30. MethodInvoker mi = new MethodInvoker(ShowProgressBar);
  31. this.BeginInvoke(mi);
  32. System.Threading.Thread.Sleep(1000); // sleep to show window
  33. for (int i = 0; i < 1000; ++i)
  34. {
  35. System.Threading.Thread.Sleep(5);
  36. // 这里直接调用代理
  37. this.Invoke(this.myHandle, new object[] { (i / 10) });
  38. }
  39. }
  40. }

子窗体代码

  1. public partial class Form2 : Form
  2. {
  3. public Form2()
  4. {
  5. InitializeComponent();
  6. }
  7. public void SetProgressValue(int value)
  8. {
  9. this.progressBar1.Value = value;
  10. this.label1.Text = "Progress :" + value.ToString() + "%";
  11. // 这里关闭,比较好,呵呵!
  12. if (value == this.progressBar1.Maximum - 1) this.Close();
  13. }
  14. }

下面是运行结果图


参考资料

1. C#进度条实现实例 { http://www.csharpwin.com/csharpspace/6546r2922.shtml }
2. 使用BackgroundWorker方便地实现多线程进度条!{ http://www.coderblog.in/2011/03/backgroundworker-for-progreessbar.html }
3. 多线程:C#.NET中使用BackgroundWorker在模态对话框中显示进度条 { http://www.mysjtu.com/page/M0/S536/536907.html } 
4. C#进度条在弹出窗口中显示的实现 { http://wenku.baidu.com/view/9f9d89d2240c844769eaeeff.html }

C#调用耗时函数时显示进度条浅探的更多相关文章

  1. idhttp post 上传或下载时显示进度条

    通过 idhttp 带进度条上传演示一下,下载和上传原理差不多,说明一下下面例子中的的idhttp 是动态创建的 第一步:添加一个StatusBar或者gauge 进度条,这2个都可以.我用的是 st ...

  2. idhttp post 上传或下载时显示进度条(对接idhttp1.OnWork事件)

    通过 idhttp 带进度条上传演示一下,下载和上传原理差不多,说明一下下面例子中的的idhttp 是动态创建的 第一步:添加一个StatusBar或者gauge 进度条,这2个都可以.我用的是 st ...

  3. Asp.Net实现无刷新文件上传并显示进度条(非服务器控件实现)(转)

    Asp.Net实现无刷新文件上传并显示进度条(非服务器控件实现) 相信通过Asp.Net的服务器控件上传文件在简单不过了,通过AjaxToolkit控件实现上传进度也不是什么难事,为什么还要自己辛辛苦 ...

  4. (委托事件处理)关于多线程执行显示进度条的实例(转)&&线程间操作无效: 从不是创建控件“rtxtEntryNO”的线程访问它。

    关于多线程执行显示进度条的实例! 之前回答了一篇关于怎么在线程中操作进度条的帖子,估计有人看的不是很明白今天没事,写了一个小小的实例,很简单,就2个文件权当抛砖引玉,希望有更好解决方案的人发表一下意见 ...

  5. Java实现在复制文件时使用进度条

    在对大文件操作时,可能会需要些时间,此时为用户提供进度条提示是非常常见的一项功能,这样用户就可以了解操作文件需要的时间信息.本实例为大家介绍了在复制大的文件时使用的进度条提示,需要注意的是,只有在读取 ...

  6. Asp.Net 无刷新文件上传并显示进度条的实现方法及思路

    相信通过Asp.Net的服务器控件上传文件在简单不过了,通过AjaxToolkit控件实现上传进度也不是什么难事,为什么还要自己辛辛苦苦来 实现呢?我并不否认”拿来主义“,只是我个人更喜欢凡是求个所以 ...

  7. VC下载文件 + 显示进度条

    在codeproject里找了许久,发现这样一个VC下载文件并显示进度条的源码,于是添加了些中文注释: 1.下载线程函数: UINT DownloadFile(LPVOID pParam) { CWn ...

  8. 基于Jquery插件Uploadify实现实时显示进度条上传图片

    网址:http://www.jb51.net/article/83811.htm     这篇文章主要介绍了基于Jquery插件Uploadify实现实时显示进度条上传图片的相关资料,感兴趣的小伙伴们 ...

  9. 【shell脚本】显示进度条

    使用动态时针版本显示进度条 [root@VM_0_10_centos shellScript]# cat progressBar.sh #!/bin/bash # 进度条,动态时针版本 # 定义显示进 ...

随机推荐

  1. 【bzoj1797】[Ahoi2009]Mincut 最小割 网络流最小割+Tarjan

    题目描述 给定一张图,对于每一条边询问:(1)是否存在割断该边的s-t最小割 (2)是否所有s-t最小割都割断该边 输入 第一行有4个正整数,依次为N,M,s和t.第2行到第(M+1)行每行3个正 整 ...

  2. bootstrap table表格属性、列属性、事件、方法

    留存一份,原文地址http://bootstrap-table.wenzhixin.net.cn/zh-cn/documentation/ 表格参数 表格的参数定义在 jQuery.fn.bootst ...

  3. 【C++ 拾遗】extern 关键字

    Separate compilation allows programs to be written in logical parts. let us split our programs into ...

  4. 2018牛客多校第一场 D.Two Graphs

    题意: n个点,m1条边的图E1,n个点,m2条边的图E2.求图E2有多少子图跟图E1同构. 题解: 用STL的全排列函数next_permutation()枚举映射.对于每一种映射枚举每一条边判断合 ...

  5. BZOJ1876 [SDOI2009]SuperGCD 【高精 + GCD优化】

    题目 Sheng bill有着惊人的心算能力,甚至能用大脑计算出两个巨大的数的GCD(最大公约 数)!因此他经常和别人比 赛计算GCD.有一天Sheng bill很嚣张地找到了你,并要求和你比 赛,但 ...

  6. CentOs7 minimal安装后没有ifconfig命令解决方法

    没有ifconfig命令目前我了解两个原因: 1./sbin/ifconfig 可以执行,但是ifconfig无法执行.这个解决的时候只需要将/sbin 添加到PATH下就可以了. 2.系统未安装if ...

  7. Spring源码解析-配置文件的加载

    spring是一个很有名的java开源框架,作为一名javaer还是有必要了解spring的设计原理和机制,beans.core.context作为spring的三个核心组件.而三个组件中最重要的就是 ...

  8. innodb_stats_on_metadata and slow queries on INFORMATION_SCHEMA

    INFORMATION_SCHEMA is usually the place to go when you want to get facts about a system (how many ta ...

  9. Codeforces ----- Kefa and Dishes [状压dp]

    题目传送门:580D 题目大意:给你n道菜以及每道菜一个权值,k个条件,即第y道菜在第x道后马上吃有z的附加值,求从中取m道菜的最大权值 看到这道题,我们会想到去枚举,但是很显然这是会超时的,再一看数 ...

  10. BZOJ 2457 双端队列(思维

    2457: [BeiJing2011]双端队列 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 582  Solved: 253[Submit][Sta ...