摘要:该有如下特点通过设置回掉处理事件,且是运行在上的通过设置间隔时间当设置为时,只在到达第一次时间间隔后触发事件是一个多线程计时器无法直接调用上的控件,需要使用委托主要用在服务中。
在C#中存在3种常用的 Timer :
System.Windows.Forms.Timer
System.Timers.Timer
System.Threading.Timer
零、System.Windows.Forms.Timer这个 Timer 是单线程的,也就是说只要它运行,其他线程就要等着。
这个 Timer 有如下特点:
完全基于UI线程,定时器触发时,操作系统把定时器消息插入线程消息队列中,调用线程执行一个消息泵提取消息,然后发送到回调方法Tick中;
使用 Start 和 Stop 启动和停止 Timer;
UI操作过长会导致 Tick 丢失;
可以使用委托Hook Tick事件;
精确度不高;
通过将 Enabled 设置为 True,使 Timer 自动运行
从上面的第一个特点可以得知,该 Timer 会造成 WinForm UI 假死,因此如果需要定时处理大量计算或者大量IO操作的任务,不建议使用该 Timer ,接下来我们看一个例子体会一下在IO操作的情况下出现的假死情况:
我们在Form中放入两个Button 一个Lable和一个Timer
private void Button_Click(object sender, EventArgs e) { timer.Interval = 1000; timer.Tick += Timer_Tick; timer.Start(); } private void Timer_Tick(object sender, EventArgs e) { for (int i = 0; i < 10000; i++) { File.AppendAllText(Directory.GetCurrentDirectory()+"test.txt", i.ToString()); this.label_output.Text = "当前操作:插入数字" + i; } }
我们单击计算按钮,我们会发现WinForm出现了假死(无法移动窗口、按钮无法点击等)
一、System.Timers.Timer该 Timer 是基于服务器的计时器,是为在多线程环境中用于辅助线程而设计的,可以在线程间移动来处理引发的 Elapsed 事件,比上一个计时器更加精确。
该 Timer 有如下特点:
通过 Elapsed 设置回掉处理事件,且 Elapsed 是运行在 ThreadPool 上的;
通过 Interval 设置间隔时间;
当 AutoReset 设置为 False 时,只在到达第一次时间间隔后触发 Elapsed 事件;
是一个多线程计时器;
无法直接调用 WinForm 上的控件,需要使用 委托;
主要用在 Windows 服务中。
同样我们通过代码来看一下该 Timer 计时器怎么使用:
System.Timers.Timer timersTimer = new System.Timers.Timer(); private void Button_Click(object sender, EventArgs e) { timersTimer.Interval = 1000; timersTimer.Enabled = true; timersTimer.Elapsed += TimersTimer_Elapsed; timersTimer.Start(); } private void TimersTimer_Elapsed(object sender, System.Timers.ElapsedEventArgs e) { for (int i = 0; i < 10000; i++) { this.BeginInvoke(new Action(() => { this.label_output.Text="当前时间:"+DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"); }), null); } } private void Button1_Click(object sender, EventArgs e) { timersTimer.Stop(); }
运行上面代码,会发现WinForm界面假死的情况消失了。
二、System.Threading.Timer该 Timer 同样也是一个多线程的计时器,它有如下特点:
多线程
和前两个计时器相比没有 Start 和 Stop 方法,如果要停止计时器,必须调用 Dispose 方法来销毁 Timer 对象;
调用 Dispose 方法后并不能马上停止所有的计时器,这是因为间隔时间小于执行时间时多个线程运行造成的,多个线程无法同时停止;
是一个轻量级的计时器;
所有的参数全部在构造函数中进行了设置;
可以设置启动时间;
不建议再 WinForm 程序中使用。
我们来看一下代码(在控制台应用程序中输入以下代码):
static System.Threading.Timer threadingTimer; static int numSum = 0; static void Main(string[] args) { threadingTimer = new System.Threading.Timer(new System.Threading.TimerCallback(threadingTimer_Elapsed), null, 0, 1000); Console.Read(); } private static void threadingTimer_Elapsed(object state) { for (int i = 0; i < 10000; i++) { numSum++; Console.WriteLine("输出数字:"+i); } if (numSum > 10000) { threadingTimer.Dispose(); Console.WriteLine("结束"); } }
注意:当我们不再需要多线程Timer计时器的时候,我们可以调用 Dispose 方法释放所占有的资源。但是因为Timer计时器是按线程池线程来安排回调执行的,因此回调可能发生在 Dispose方法的重载被调用之后,所以我们可以使用可使用 Dispose(WaitHandle) 方法等待所有回掉完成。
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/61649.html
摘要:该有如下特点通过设置回掉处理事件,且是运行在上的通过设置间隔时间当设置为时,只在到达第一次时间间隔后触发事件是一个多线程计时器无法直接调用上的控件,需要使用委托主要用在服务中。 在C#中存在3种常用的 Timer : System.Windows.Forms.Timer System.Timers.Timer System.Threading.Timer 零、System.Wind...
摘要:但是实际想想里面的属性跟也没有什么区别,他们的可访问性跟是一样的如果是全局,里面的属性也是全局的要改里面的随便改改。在怎样,还有的同学熟悉不,类的继承,面向对象一应俱全,哈哈,当然编译出来的是这样的装逼结束,收工 javascript是脚本语言?javascript万物皆对象? 呵呵哒!带你见识下js面对对象的魅力 是的是的,退后,朕要开始装逼了~ showImg(https://se...
摘要:但是实际想想里面的属性跟也没有什么区别,他们的可访问性跟是一样的如果是全局,里面的属性也是全局的要改里面的随便改改。在怎样,还有的同学熟悉不,类的继承,面向对象一应俱全,哈哈,当然编译出来的是这样的装逼结束,收工 javascript是脚本语言?javascript万物皆对象? 呵呵哒!带你见识下js面对对象的魅力 是的是的,退后,朕要开始装逼了~ showImg(https://se...
摘要:但是实际想想里面的属性跟也没有什么区别,他们的可访问性跟是一样的如果是全局,里面的属性也是全局的要改里面的随便改改。在怎样,还有的同学熟悉不,类的继承,面向对象一应俱全,哈哈,当然编译出来的是这样的装逼结束,收工 javascript是脚本语言?javascript万物皆对象? 呵呵哒!带你见识下js面对对象的魅力 是的是的,退后,朕要开始装逼了~ showImg(https://se...
摘要:一个写的接收电子称数据并解析出重量数据的类。之前单位购买了两架电子称,同时单独购买了配套的软件。电子称参数类有些电子称发送的数据格式是没有小数点分隔符例如耀华没有小数点的数中整数部分长度数据接收处理类比较简单,请参考注释即可。 ...
阅读 2001·2023-04-25 19:06
阅读 1216·2021-11-17 09:33
阅读 1634·2019-08-30 15:53
阅读 2413·2019-08-30 14:20
阅读 3431·2019-08-29 12:58
阅读 3394·2019-08-26 13:27
阅读 363·2019-08-26 12:23
阅读 361·2019-08-26 12:22