C# 等待框----Application.DoEvents()实现

 转自:http://hi.baidu.com/sageking2/blog/item/67a8bf0819b2ec9d0b7b820c.html

 于Application.DoEvents()的小研究

在MSDN中的备注是:

当运行 Windows 窗体时,它将创建新窗体,然后该窗体等待处理事件。该窗体在每次处理事件时,均将处理与该事件关联的所有代码。所有其他事件在队列中等待。在代码处理事件时,应用程序并不响应。例如,当将另一窗口拖到该窗口前面时,该窗口不重新绘制。

如果在代码中调用 DoEvents,则您的应用程序可以处理其他事件。例如,如果您有向 ListBox 添加数据的窗体,并将 DoEvents 添加到代码中,那么当将另一窗口拖到您的窗体上时,该窗体将重新绘制。如果从代码中移除 DoEvents,那么在按钮的单击事件处理程序执行结束以前,您的窗体不会重新绘制。

刚开始接触这段话的时候,会读不懂。所以我就去找了篇文章,文章中通过一个循环输出文本的例子解释了通过这段代码可以给人带来很好的用户体验( 不加的话程序只会显示输出循环最后的结果,但是加了以后会实时的输出),从而消除了认为进程死掉的情况。他还提到了用这个的缺点是影响了进程的效率,并通 过了一个测验证实他的观点。

看了他的博客,我也就仿照他的说明写了个代码进行了测验,第一个代码是同他说的一样,加不加代码的效果是不同的。但是第二个,因为用一个计时器来计时。我因为不是很懂计时器,我就写了个类似计时器的代码,原理感觉上和他的差不多。

原理是:用Timer组件,写了一个Tick事件并设置了间隔时间,在事件内用了一个counter 来计数,再通过计数的结果已经响应的时间间隔来计算时间。再运行加了DoEvents代码的运行的很好,和我想的一样。

然后我又写了一个

       运行后发现,textBox2.可以得到99999就是结果的值,但是 label2却一直是零,也就相当于没有触发Tick事件,这就把给搞昏了,我明明是设定了Timer的Start而且也在循环之前调用的Tick 事件,这是为什么呢?我又试着将代码里的stop 给注释掉,再运行,在循环结束后可以运行Tick 事件了。从这个结果来看,开始以为Tick本来是可以运行的。因为Tick的触发是在当指定的计时器间隔已过去而且计时器处于启用状态时发生。那么有可能 时间间隔太长,导致时间间隔没解释,也就是Tick来不及触发。为了验证这个是否正确,我将时间间隔调小,但是结果还是那样,而且到后来我直接将Tick 的事件触发代码带到了循环中,可结果还是那样。到此,我想也许是存在优先级的概念,是否循环这个进程的优先级比Tick的优先级高。之前在循环体内设置的 到99999后的Stop掉Timer,直接导致Tick刚要执行就被强行关掉了。因为MSDN上也查不到这方面的原因,我就此当做是结论了。

 

      从这个测验后再回头想DoEvents的功能,我就想DoEvents就好比实现了进程的同步。在不加的时候,因为优先级的问题,程序会执行主进程的代码,再执行Tick的代码,而加了以后就可以同步执行。

从这个测验后再回头想DoEvents的功能,我就想DoEvents就好比实现了进程的同步。在不加的时候,因为优先级的问题,程序会执行主进程的代码,再执行Tick的代码,而加了以后就可以同步执行。

再回想那位网友在博客里写到的:

http://www.cnblogs.com/datong/archive/2008/04/06/1139216.html

记得第一次使用Application.DoEvents()是为了在加载大量数据时能够有一个数据加载的提示,不至于系统出现假死的现象,当时也 没有深入的去研究他的原理是怎样的,结果在很多地方都用上了Application.DoEvents(),今天看到了关于这方面的一些文章,知道我以前 有些用法是不当的,有些地方需要慎用Application.DoEvents()。
首先我们先看看在循环比较大的程序中,它的作用还是不错的,起到了一个实时响应的效果,例如:

for (int q = 0; q < 1000000; q++)
             {
                 textBox1.Text = q.ToString();
                 Application.DoEvents();//实时响应文本框中的值
             }

如果没有加上 DoEvents的话,由于循环时间会比较久就会出现假死的状态,而且程序不能处理其他的事件。而如果加上DoEvents的话就会对文本框的值实时响 应,给用户带来较好的用户体验,可是DoEvents也带来了效率上的问题,处理同样的一个事件调用了DoEvents后效率降低了好几倍,这也是为什么 要慎用的原因了。

现在来分析的话,就可以解释为什么,加了这行代码后。程序就会出实现类似双线程的功能,他会去寻找是否有别的事件,因为有这段时间才使得数值有时间可以显现。

1楼:Windows是消息驱动的,DoEvents就是让程序去执行还没有处理的消息。处理消息带来效率的下降那是必然的。刷新界面其实也是一种消息。

如果循环量很大,比较好的方法是处理过程在独立的线程中运行,而界面最好每隔一定时间再刷新一次。

Form:http://www.cnblogs.com/jianggest/archive/2009/04/18/1437513.html 

Code
1 private void button2_Click(object sender, EventArgs e)
2
3          {
4
5             this.timer1.Start();
6
7             this.timer1.Interval = 1000;
8
9             this.timer1.Tick += new EventHandler(timer1_Tick);
10
11             for (int q = 0; q < 100000; q++)
12
13              {
14
15                  textBox2.Text = q.ToString();
16
17                 if (q == 99999)
18
19                  {
20
21                     this.timer1.Stop();
22
23                    }
24
25              }           
26
27             label2.Text = num.ToString ();
28
29

Code
1 private void button1_Click(object sender, EventArgs e)
2          {
3             this.timer1.Start();
4             this.timer1.Interval = 1000;
5             this.timer1.Tick += new EventHandler(timer1_Tick);
6             for (int q = 0; q < 10000; q++)
7              {
8                  textBox1.Text = q.ToString();
9                  Application.DoEvents();
10                 if (q == 9999)  
11                  {
12                     this.timer1 .Stop ();
13                    }
14              }
15          }
16
17 private void timer1_Tick(object sender, EventArgs e)
18          {
19              num++;
20             label2.Text = num.ToString();
21            //MessageBox.Show("hehe!");
22          }
23

呵呵,关键自己程序里要用到 所以开发了这个小功能 很多地方很粗糙,俺菜鸟,高手们就别贬我了。 使用的时候把2个dll放到你的程序目录,在资源管理器引用LOADing.dll 就可以了,DevComponents.DotNetBar2.dll为确定按钮控件的引用 列子: private void dl_Click(object sender, EventArgs e) { LOADing.FORMshow load = new LOADing.FORMshow(); load.showto(this, delegate { hand(new object[] { load, "正在处理数据" }); },false); } private void hand(object fr) { int i = 0; while (i < 100) { i++; ((LOADing.FORMshow)((object[])(fr))[0]).send((string)((object[])(fr))[0]+i.ToString()); System.Threading.Thread.Sleep(100); } } 主要用于处理数据的时候,提示用户处理过程,防止界面假死,数据处理完毕后会自动关闭窗体。 注:this为所要调用等待窗体的主窗体对象,中间为数据传递的委托,显示数据处理的过程.load.showto(this, delegate { hand(new object[] { load, "正在处理数据" }); });中new object[] 第一个参数一定要为固定的参数:创建LOADing.FORMshow的实例,后面再就可跟任意数据,都可在方法的过程中调用显示,最后的bool参数:false方法函数执行完毕后自动关闭窗体显示;true为方法执行完毕后出现确定按钮并阻塞主线程UI,点击确定后关闭提示窗体并取消阻塞线程,这么简单,用相信大家都会用了。 界面没有进行美化,感觉这样的就可以了,随后会升级为可自定义界面! 有问题加我QQ76230454
与前版功能基本相似,修正了部分bug,对界面进行了美化,目前这个美化相信应该够用了,因为时间问题没有增加可自定义界面功能,等有时间再提供吧,压缩文件使用“好压”做的,里面有一个例子,图片资源编辑工具,大家可以测试一下效果,里面也有详细说明,图片资源编辑器大家也可以用到自己的程序中,方便程序的图片统一管理和调用,菜鸟,达人们别笑话我了。 这里还是在说一下等待窗体的具体使用方法吧 首先将LOADing.dll,DevComponents.DotNetBar2.dll两个dll文件复制到你程序目录中,在程序项目中引用LOADing.dll,在要使用的地方 //先实例 LOADing.FORMshow FRload = new LOADing.FORMshow(); //再调用showto方法,其中的参数this为你调用等待窗体的主窗体对象,delegate { }为委托,IMGclass_AddFlie_r()为功能处理函数,其中所传递的参数第一的FRload必须为固定的创建等待窗体的实例对象,后面跟所需要传递的对象参数。 FRload.showto(this, delegate { IMGclass_AddFlie_r(new object[] { FRload, iclass, fileDialog1.FileNames, _at.SelectedNode.Text }); },true); //这个为数据处理部分 private void IMGclass_AddFlie_r(object[] d) { for (int i = 0; i <= ((string[])d[2]).Length - 1; i++) { ((IMGclass)d[1]).top[d[3].ToString()].Add("标" + ((IMGclass)d[1]).top[d[3].ToString()].Count, BinToCmd(((string[])d[2])[i])); f_new_hand(new object[] { ((IMGclass)d[1]).top[d[3].ToString()], "标" + (((IMGclass)d[1]).top[d[3].ToString()].Count - 1) }); ((LOADing.FORMshow)d[0]).send("加载图片文件:", Convert.ToInt32((Convert.ToSingle(i) / (Convert.ToSingle(((string[])d[2]).Length) / Convert.ToSingle(100))))); } BinToclass(((IMGclass)d[1]), _path[_at.SelectedNode.Parent.Text]); } 好了,使用起来很简单,看看上面的例子就会了,如需索要源码或者要提问的话,请联系QQ76230454.
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值