时间:2021-05-20
不知道你是否对.NET里面的定时器产生过一些疑问,以下是武小栈个人的一些总结。
在.NET的框架之内定时器有四种,先看一下微软官方对他们各自特点介绍:
再看看微软对开发者的使用建议:
System.Threading.Timer 是一种简单的轻型计时器,它使用回调方法,并由线程池线程提供服务。 不建议与 Windows 窗体一起使用,因为它的回调不会在用户界面线程上发生。 System.Windows.Forms.Timer 是用于 Windows 窗体的更好选择。 对于基于服务器的计时器功能,您可以考虑使用 System.Timers.Timer,这会引发事件并具有其他功能。
是一个基础类,使用起来不是太好用,各种用法较为原始,用的较少。
第一次接触的就是它,毕竟直接winform拖下来就行了,用的还是比较多,我通常用在运行一些刷新界面的代码,这些代码通常不会有什么逻辑运算,比如界面上需要显示一个倒计时。
在这个类使用中我遇到过两个疑惑,作为分享:
Q1:Tick实践会创建新线程执行吗?
A1:不会创建新的线程,始终在主线程里面运行Tick事件;
Q2:定时器会start()瞬间触发一次,还是等待Interval间隔后再触发?
A2:等待Interval间隔后再触发。
Q3:定时器start()和stop()时候Interval会累积吗?
A3:不累积,每次start()重新计时。
Q4:如果Tick事件内的代码未执行完成,但是下一次Tick定时已经达到会发生什么?
A4:不会强行终止未完成的代码,也不会因为上一次Tick事件代码未执行完成而不再触发,而是类似于栈的形式将之前未执行完成的代码堆积,后触发的Tick事件内的代码先执行,先触发未完成的代码后执行,具体可以看下面示例。
public Form1() { InitializeComponent(); timerForm.Tick += TimerForm_Tick; } private int num = 1;//一个序号,表示当前第几次进入Tick事件 private int rowNum = 1;//一个全局的行号,记录一下总共AppendText多少次 private void TimerForm_Tick(object sender, EventArgs e) { string s = $"我是第{num++}次"; for (int i = 0; i < 5; i++) { textBox1.AppendText($"{rowNum++} {s} 序号i={i} 当前线程ID={Thread.CurrentThread.ManagedThreadId.ToString()} \r\n"); Delay(1000); } } private Timer timerForm = new Timer(){Interval = 1000}; private void button1_Click(object sender, EventArgs e) { textBox1.AppendText("button " + Thread.CurrentThread.ManagedThreadId.ToString() + "\r\n"); timerForm.Start(); } public static void Delay(int mimillisecond) { int start = Environment.TickCount; while (Math.Abs(Environment.TickCount - start) < mimillisecond) { System.Windows.Forms.Application.DoEvents(); } }是对System.Threading.Timer的一层封装,都是通过委托方法TimerCallback进行回调触发定时器事件,可以先看看System.Timers.Timer的代码实现方式:
if (!value) { if (this.timer != null) { this.cookie = (object) null; this.timer.Dispose(); this.timer = (System.Threading.Timer) null; } this.enabled = value; } else { this.enabled = value; if (this.timer == null) { if (this.disposed) throw new ObjectDisposedException(this.GetType().Name); int dueTime = (int) Math.Ceiling(this.interval); this.cookie = new object(); this.timer = new System.Threading.Timer(this.callback, this.cookie, dueTime, this.autoReset ? dueTime : -1); } else this.UpdateTimer(); }不过 System.Threading.Timer的属性和方法都更加友善,我通常在使用中不设计更新界面,都会使用这个定时器类,有一点要说明的是,将SynchronizingObject属性赋值到控件后,事件中代码会在控件上委托调用,如timer.SynchronizingObject = this;可以看下System.Timers.Timer内部是如何实现的。
if (elapsedEventHandler != null) { if (this.SynchronizingObject != null && this.SynchronizingObject.InvokeRequired) { this.SynchronizingObject.BeginInvoke(elapsedEventHandler, new object[] { this, elapsedEventArgs }); } else { elapsedEventHandler(this, elapsedEventArgs); } }虽然System.Timers.Timer定时器理论上是不受单线程限制,可以短时间内触发多次,但是实际上会受到线程池的限制,先看巨硬对于此的说明:
如果 nullSynchronizingObject 属性,则在 ThreadPool 线程上引发 Elapsed 事件。 如果 Elapsed 事件的处理持续时间超过 Interval,则可能会在其他 ThreadPool 线程上再次引发该事件。 在这种情况下,事件处理程序应该是可重入的。
1、当SynchronizingObject不为null,将在指定的对象线程上触发事件,为单线程触发,与System.Windows.Forms.Timer执行方式相同;
2、当SynchronizingObject不为null时将在线程池(ThreadPool)上引发事件,执行事件内的代码。理论上可以重复载入,但是会受到ThreadPool线程数限制,比如ThreadPool.SetMaxThreads(8, 8),那么定时器触发事件只能同时载入8次;
我现在用定时器基本上都是用System.Timers.Timer,在我看来System.Timers.Timer可以用SynchronizingObject属性实现在主线程运行,也可以不设置SynchronizingObject属性,是事件在线程池里触发,作为后台线程使用,基本能满足我在开发中的使用需求。
System.Timers Namespace
System.Windows.Forms
System.Threading.ThreadPool Class
以上就是c# 区分几种定时器(timer)的详细内容,更多关于c# 定时器(timer)的资料请关注其它相关文章!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
本文实例讲述了C#中timer定时器用法。分享给大家供大家参考。具体如下:下面的代码通过Timer定时器每隔1000毫秒(1秒)触发一次事件usingSyste
本文实例讲述了C#多线程学习之使用定时器进行多线程的自动管理。分享给大家供大家参考。具体分析如下:Timer类:设置一个定时器,定时执行用户指定的函数。定时器启
C#中timer类的用法关于C#中timer类在C#里关于定时器类就有3个1.定义在System.Windows.Forms里2.定义在System.Threa
本文实例讲述了C#使用timer定时在屏幕上输出信息的方法。分享给大家供大家参考。具体分析如下:这段c#代码通过timer定时器每隔5秒钟调用一次OnTimer
关于C#Timer类在C#里关于定时器类就有3个C#Timer使用的方法1.定义在System.Windows.Forms里C#Timer使用的方法2.定义在S