.NET中有多少种定时器
.NET中至少有6种定时器,每一种定时器都有它的用途和特点。根据定时器的应用场景,可以分为UI相关的定时器和UI无关的定时器。本文将简单介绍这6种定时器的基本用法和特点。
UI定时器
.NET中的UI定时器主要是WinForm、WPF以及WebForm中的定时器。分别为:
System.Windows.Forms.Timer
System.Windows.Threading.DispatcherTimer
System.Web.UI.Timer
通常情况下,WinForm、WPF中的定时器是在UI线程上执行回调函数,因此可以直接访问UI元素。由于WinForm、WPF支持单线程单元模型(Single-Thread Apartment,STA),定时器间隔事件是在UI线程上触发,因此,不用担心线程安全问题。
System.Web.UI.Timer
是通过Javascript定时器和服务端异步回调实现,也是单线程的。
请注意,这里说的是通常情况,后边介绍
System.Windows.Threading.DispatcherTimer
时会提到在非UI线程创建DispatcherTimer
时也无法直接访问UI元素。
System.Windows.Forms.Timer
System.Windows.Forms.Timer
针对WinForm应用进行了优化,是只能在WinForm上使用的定时器。这个定时器是针对单线程环境设计的,是在UI线程上处理定时任务。
它要求用户代码有可用的UI消息泵,定时任务须在UI线程上运行,或者跨线程通过Invoke
或者BeginInvoke
封送(marshal)到UI线程上运行。其优点是使用简单,只需通过给Interval
属性赋值来设置时间间隔,并注册Tick
事件处理定时任务。其缺点是精度不高,精度为55毫秒,也就是Interval
赋值小于55时,也是55毫秒触发一次定时任务。
public partial class TimerFrom : Form
{
private System.Windows.Forms.Timer digitalClock;
private void TimerFrom_Load(object sender, EventArgs e)
{
digitalClock = new System.Windows.Forms.Timer();//创建定时器
digitalClock.Tick += new EventHandler(HandleTime);//注册定时任务事件
digitalClock.Interval = 1000;//设置时间间隔
digitalClock.Enabled = true;
digitalClock.Start(); //开启定时器
}
public void HandleTime(Object myObject, EventArgs myEventArgs)
{
labelClock.Text = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss");
}
private void frmTimerDemo_FormClosed(object sender, FormClosedEventArgs e)
{
digitalClock.Stop();//停止定时器
digitalClock.Dispose();
}
}
System.Windows.Threading.DispatcherTimer
System.Windows.Threading.DispatcherTimer
是WPF中的定时器,它是基于Dispatcher
对象的(并不是基于UI线程的)。DispatcherTimer
的定时任务是像其他操作一样放在Dispatcher
队列上,其执行操作时间依赖于队列中其他任务及其优先级,因此,DispatcherTimer
不保证在时间间隔发生时准确执行,只保证不会在时间间隔发生前执行。
Dispatcher
为特定线程维护工作项(操作)的优先级队列,在线程上创建Dispatcher
对象时,它成为唯一可以关联该线程的Dispatcher
对象,WPF中,DispatcherObject
只能被与之关联的Dispatcher
对象访问,也就是非UI线程中无法直接访问UI元素(WPF中的UI元素都是派生自DispatcherObject
)
此外,DispatcherTimer
不像System.Windows.Forms.Timer
那样只在UI线程上创建才能触发Tick
事件,它在非UI线程下创建也可以触发Tick
事件,此时访问UI元素也需要通过Invoke
或者BeginInvoke
封送(marshal)到UI线程上运行。其优点也是简单易用,适合在UI线程上执行任务或触发事件,缺点是精度不准确,可能存在延迟。
private void Dt_Tick(object sender, EventArgs e)
{
Dispatcher.BeginInvoke((Action)delegate ()
{
text1.Text = DateTime.Now.ToString();
});
Console.WriteLine(DateTime.Now.ToString());
}
private void Button_Click(object sender, RoutedEventArgs e)
{
Task.Run(() =>{
DispatcherTimer dt = new DispatcherTimer();
dt.Tick += Dt_Tick;
dt.Interval = TimeSpan.FromSeconds(1);
dt.Start();
Dispatcher.Run();
});
}
上述代码中,DispatcherTimer
是非UI线程中创建,定时任务中访问UI元素text1,需要通过Invoke
或者BeginInvoke
封送(marshal)到UI线程上运行,而Console.WriteLine
则可以直接运行。
System.Web.UI.Timer
System.Web.UI.Timer
是仅适用于.NET Framework
的ASP.NET
组件。通过Javascript定时器和服务端异步回调实现。每次触发定时器时,只能执行一个异步回调方法,而其他的异步回调方法需要等待前一个异步回调方法执行完毕后才能执行。这样可以保证在任意时刻只有一个异步回调方法在执行,避免了多线程并发执行的问题。
UI无关定时器
从 .NET 6开始,UI无关定时器有三个:
System.Threading.Timer
System.Timers.Timer
System.Threading.PeriodicTimer
(.NET 6+)
System.Threading.Timer
System.Threading.Timer
是最基础轻量的定时器,它将定期在线程池线程上执行单个回调方法。在创建定时器对象时必须指定回调方法,并且后续不能修改,同时也可以指定定时器回调开始执行的时间以及时间间隔。定时器创建后可以通过Change
方法修改回调开始执行的时间以及时间间隔。该定时器的优点是轻量,精度相对较高,与Windows操作系统时钟精度一致,大约15毫秒。但因为是基于线程池的,所以在任务执行时间较长或者线程池过载时,会出现延迟。其缺点是使用不太方便,定时器创建后无法修改回调方法。
var stateTimer = new
var autoEvent = new AutoResetEvent(false);
Timer(CheckStatus, autoEvent, 1000,250);
private int invokeCount=0;
public void CheckStatus(Object stateInfo)
{
AutoResetEvent autoEvent = (AutoResetEvent)stateInfo;
Console.WriteLine("{0} Checking status {1,2}.",DateTime.Now.ToString("h:mm:ss.fff"),(++invokeCount).ToString());
if(invokeCount == 10)
{
invokeCount = 0;
autoEvent.Set();
}
}
System.Timers.Timer
System.Timers.Timer
在内部使用System.Threading.Timer
,并公开了更多的属性,如AutoReset
, Enabled
或SynchronizingObject
,这些属性允许配置回调的执行方式。此外,Tick事件允许注册多个处理程序。因此,一个定时器可以触发多个处理程序。还可以在计时器启动后更改处理程序。与System.Threading.Timer
相似,其优点也是精度相对较高,与Windows操作系统时钟精度一致,大约15毫秒。因为默认(或者SynchronizingObject=null
时)是基于线程池的,所以在任务执行时间较长或者线程池过载时,会出现延迟。但使用要更简便一些。
public partial class TimerFrom : Form
{
private System.Timers.Timer timer;
private void TimerFrom_Load(object sender, EventArgs e)
{
// 支持注册多个处理程序
timer.Elapsed += (sender, e) => { label1.Text = DateTime.Now.ToLongTimeString(); };
timer.Elapsed += (sender, e) => { Console.WriteLine(DateTime.Now.ToLongTimeString()); };
//自定义回调执行的方式(指定对象所在的线程),SynchronizingObject=null时在线程池上执行
timer.SynchronizingObject = this;
timer.AutoReset = true;
timer.Start();
}
}
本例中将SynchronizingObject
属性设置为Form
对象,因此Elapsed
的处理程序在UI线程上执行,可以直接修改 label1.Text
,如果SynchronizingObject
属性为null
,处理程序则是在线程池线程上执行,修改 label1.Text
时需要通过Invoke
或者BeginInvoke
封送(marshal)到UI线程上运行。
System.Threading.PeriodicTimer
System.Threading.PeriodicTimer
是 .NET 6中引入的定时器。它能方便地使用异步方式,它没有Tick
事件,而是提供WaitForNextTickAsync
方法处理定时任务。通常是使用While
循环结合CancellationToken
一起使用。和CancellationToken
一起用的时候需要注意,如果CancellationToken
被取消的时候会抛出一个OperationCanceledException
需要考虑自己处理异常。相比之前的定时器来说,有下面几个特点:[1]
- 没有
callback
来绑定事件; - 不会发生重入,只允许有一个消费者,不允许同一个
PeriodicTimer
在不同的地方同时WaitForNextTickAsync
,不需要自己做排他锁来实现不能重入; - 异步化。之前的 timer 的 callback 都是同步的,使用新 timer 可以使用异步方法,避免了编写 Sync over Async 代码;
- Dispose 之后,实例就无法使用,并且 WaitForNextTickAsync 始终返回 false。
var cts = new CancellationTokenSource(TimeSpan.FromSeconds(15));
using (var timer = new PeriodicTimer(TimeSpan.FromSeconds(1)))
{
try
{
while (await timer.WaitForNextTickAsync(cts.Token))
{
await Task.Delay(3000);
Console.WriteLine($"ThreadId is {Thread.CurrentThread.ManagedThreadId} --- Time is {DateTime.Now:HH:mm:ss}");
}
}
catch (OperationCanceledException)
{
Console.WriteLine("Operation cancelled");
}
}
小结
我们在开发过程中遇到的坑往往不是技术本身的坑,而是我们滥用没有掌握的技术导致的,在有多种技术方案可选的时候,通常只关注技术的优点,忽略了技术适用场景及其局限性。.NET中几种定时器各自都有其适用场景和不足,但都不支持高精度计时。了解这些有助于我们在开发过程中选择合适定时器,避免遇到问题后被动地替换。
.NET中有多少种定时器的更多相关文章
- wpf中应该使用c#四种定时器中的DispatcherTimer
c#中有四种定时器 1:System.Threading.Timer 使用: private System.Threading.Timer timerClose; timerClose = new S ...
- Objective-C三种定时器CADisplayLink / NSTimer / GCD的使用
OC中的三种定时器:CADisplayLink.NSTimer.GCD 我们先来看看CADiskplayLink, 点进头文件里面看看, 用注释来说明下 @interface CADisplayLin ...
- 动手学习TCP:4种定时器
上一篇中介绍了TCP数据传输中涉及的一些基本知识点.本文让我们看看TCP中的4种定时器. TCP定时器 对于每个TCP连接,TCP管理4个不同的定时器,下面看看对4种定时器的简单介绍. 重传定时器使用 ...
- TCP四种定时器--学习笔记
TCP使用四种定时器: 重传定时器(Retransmission Timer).坚持定时器(Persistent Timer).保活定时器(Keeplive Timer).时间等待定时器(Time_W ...
- javascript两种定时器的使用及其清除
<!--示例代码如下:--><!DOCTYPE html> <html> <body> <p>A script on this page s ...
- Java语言中有4种访问修饰符
转载:http://wuhaidong.iteye.com/blog/851754 Java语言中有4种访问修饰符 在Java语言中有4中访问修饰符:package(默认).private.publi ...
- 从51跳新唐cortex学习3——细说新唐两种定时器
的定时器我们已经是相当熟悉啦.但是,对于第一次接触新唐,第一次接触cortex的定时器一样,都是加1计数,但是功能和容量变大了),包括4个32位定时器(里面放24位的数据),还有分频功能.TMER0到 ...
- Qt分析:Qt中的两种定时器(可是QObject为什么要提高定时器呢,没必要啊。。。)
Qt有两种定时器,一种是QObject类的定时器,另一种是QTimer类的定时器. (1)QObject类的定时器 QObject类提供了一个基本的定时器,通过函数startTimer()来启 ...
- cocos2dx三种定时器使用
cocos2dx三种定时器的使用以及停止schedule.scheduleUpdate.scheduleOnce 今天白白跟大家分享一下cocos2dx中定时器的用法. 首先,什么是定时 ...
- java中有几种类型的流?JDK为每种类型的流提供了一些抽象类以供继承,请说出他们分别是哪些类?
java中有几种类型的流?JDK为每种类型的流提供了一些抽象类以供继承,请说出他们分别是哪些类? Java中的流分为两种,一种是字节流,另一种是字符流,分别由四个抽象类来表示(每种流包括输入和输出两种 ...
随机推荐
- PHP秒杀面试题
什么是秒杀系统:秒杀系统是一个处理大量并发用户请求的系统,通常用于限时促销或特定活动中,用户可以在特定时间内以抢购的方式购买商品或服务. 秒杀系统可能面临的挑战是什么?秒杀系统可能面临以下挑战: 高并 ...
- 操作系统实验——利用Linux的消息队列通信机制实现两个线程间的通信
目录 一. 题目描述 二.实验思路 三.代码及实验结果 四.遇到问题及解决方法 五.参考文献 一. 题目描述 编写程序创建三个线程:sender1线程.sender2线程和receive线程,三个线程 ...
- virtualbox克隆虚拟机
1.选择要克隆的虚拟机 2.设置克隆机的名称和存放位置 3.选择克隆类型 4.克隆结果
- Python Web:Django、Flask和FastAPI框架对比
Django.Flask和FastAPI是Python Web框架中的三个主要代表.这些框架都有着各自的优点和缺点,适合不同类型和规模的应用程序. Django: Django是一个全功能的Web框架 ...
- Programming abstractions in C阅读笔记:p130-p131
<Programming Abstractions In C>学习第52天,p130-p131,总结如下: 一.技术总结 1. pig latin game 通过pig latin gam ...
- 图解 LeetCode 算法汇总——二分查找
二分查找(Binary Search)是一种在有序数组中查找特定元素的高效算法.它的基本思想是将目标值与数组中间的元素进行比较,如果目标值小于中间元素,则在数组的左半部分继续查找,否则在右半部分查找, ...
- 逻辑漏洞挖掘之XSS漏洞原理分析及实战演练
一.前言 2月份的1.2亿条用户地址信息泄露再次给各大公司敲响了警钟,数据安全的重要性愈加凸显,这也更加坚定了我们推行安全测试常态化的决心.随着测试组安全测试常态化的推进,有更多的同事对逻辑漏洞产生了 ...
- linux中的sar命令
linux中的sar命令 sar命令的安装 [root@localhost test]# yum install sysstat 安装成功! sar命令说明 语法格式 sar [ 选项 ] [ < ...
- Informix 4gl错误代码信息和更正
(一)Informix信息和更正 出版日期:6 1996 年 年 1 11 月 0 成功. 操作成功.当 SQL 语句成功地执行时,数据库服务器把这个 SQLCODE 值返回给应用程序. 100 没有 ...
- 将GitBash设置为VS Code的默认终端
这个东西搞了半天,真的无语...网上的东西都太旧了 注意:"terminal.integrated.shell.windows"自2021年4月起已弃用. 1.首先打开设置 2.进 ...