C#实现1ms定时器不精准?如何实现一个高性能高精度的1ms定时器?(附完整示例Demo)
在C#日常开发中,我们经常需要使用定时器(Timer)进行周期性任务的执行。
例如,每隔1秒打印一条日志,或每隔100毫秒执行某个数据刷新逻辑。
但是,当我们尝试在C#中实现一个1毫秒(1ms)级别的高精度周期性定时时,往往会发现结果并不如预期理想。
标准的托管定时器(如 System.Threading.Timer
或 System.Timers.Timer
)虽然能简单快速地实现周期性调用,但在1ms级别的精度要求下却经常出现几ms延迟甚至几十ms的偏差。
测试下来是和电脑的性能会有关系。
本文将深入分析为什么在C#中难以实现精确的1ms定时,并给出解决方案。
我们将利用 Windows 的多媒体定时器(Multimedia Timer)来显著提高定时精度,并给出完整的示例代码,让您在实际项目中轻松上手。
为什么C#的托管定时器不精确?
1. 操作系统定时器分辨率的限制
Windows系统本身对计时有一定的粒度限制。默认情况下,Windows的系统计时器频率可能在10~15.6ms左右。当您使用标准托管定时器设定间隔为1ms时,其实是向操作系统提出了一个“请求”,但由于底层定时器分辨率所限,最终触发往往会推迟到下一个系统计时点,导致数毫秒到十几毫秒的延迟。这也意味着在1秒内,您可能只会收到数百次回调,而非理想中的1000次。
2. 托管环境和垃圾回收(GC)
C# 运行在 .NET CLR(或 .NET Runtime)之上,这是一层托管运行环境。托管运行时需要进行垃圾回收、线程调度和JIT编译等工作。这些过程会引入额外的延迟和不确定性,使定时器触发时间更加不可控。在满负载或GC频繁触发时,定时回调的执行可能被进一步推迟。
3. 线程池和任务调度
System.Threading.Timer
和 System.Timers.Timer
通常基于线程池调度任务。线程池不是为毫秒级别的实时触发而设计的,它更倾向于在较低精度要求下的周期性或延迟任务。因此,即使设定了1ms的间隔,实际触发时刻仍会受到线程池任务排队和分配的影响。
如何提升定时精度?
若要实现接近1ms级别的定时精度,需要采取以下策略:
-
提高系统定时器精度:
使用timeBeginPeriod(1)
来设置系统全局计时器分辨率为1ms。此举会让Windows的定时粒度更精细,让后续的定时调用有更高概率以1ms级别进行触发。需要注意,提高系统计时精度会增加系统整体负载与耗电。 -
使用多媒体定时器(timeSetEvent):
多媒体定时器是Windows早期为音视频播放等对时序要求较高的场景而设计的API。与标准托管定时器相比,多媒体定时器能在极短的间隔内更加稳定地触发回调,从而提升定时精度。 -
使用原子操作和低延迟回调:
在回调中使用Interlocked
系列函数来确保多线程场景下的计数操作是线程安全和低开销的。同时避免在回调中执行过于复杂的逻辑,将回调尽可能简化以缩短执行时间,降低后续触发的抖动。 -
考虑实时系统或硬件计时(如果要求更高):
当需要真正严格的实时性能(如工业控制或设备驱动开发),Windows并非最佳平台。这时应考虑实时操作系统(RTOS)、专用硬件计数器或微控制器,以达到真正的毫秒乃至微秒级精度。
使用多媒体定时器的完整示例代码
下面的代码展示了如何使用多媒体定时器 timeSetEvent
来实现一个1ms定时器和一个1s定时器:
- 1ms定时器:每1ms增加一次计数器
_counter
- 1s定时器:每1秒读取
_counter
值并打印,然后重置为0
using System;
using System.Diagnostics;
using System.Runtime.InteropServices;
using System.Threading;class Program
{[DllImport("winmm.dll", SetLastError = true)]private static extern uint timeSetEvent(uint uDelay,uint uResolution,TimeProc lpTimeProc,IntPtr dwUser,uint fuEvent);[DllImport("winmm.dll", SetLastError = true)]private static extern uint timeKillEvent(uint uTimerId);[DllImport("winmm.dll", SetLastError = true)]private static extern int timeBeginPeriod(uint uMilliseconds);[DllImport("winmm.dll", SetLastError = true)]private static extern int timeEndPeriod(uint uMilliseconds);private delegate void TimeProc(uint uTimerID, uint uMsg, IntPtr dwUser, IntPtr dw1, IntPtr dw2);private static uint _timerId1ms;private static uint _timerId1s;private static long _counter = 0;private static TimeProc _timeProc1ms;private static TimeProc _timeProc1s;static void Main(){// 提高系统定时器精度到1mstimeBeginPeriod(1);_timeProc1ms = new TimeProc((uTimerID, uMsg, dwUser, dw1, dw2) =>{// 每1ms增加一次计数Interlocked.Increment(ref _counter);});_timeProc1s = new TimeProc((uTimerID, uMsg, dwUser, dw1, dw2) =>{// 每1s读取并打印计数long currentCount = Interlocked.Read(ref _counter);Debug.WriteLine($"当前计数值: {currentCount}");Interlocked.Exchange(ref _counter, 0);});// 启动1ms定时器(周期性)_timerId1ms = timeSetEvent(1, 0, _timeProc1ms, IntPtr.Zero, 1);if (_timerId1ms == 0){Console.WriteLine("1ms定时器创建失败");Cleanup();return;}// 启动1s定时器(周期性)_timerId1s = timeSetEvent(1000, 0, _timeProc1s, IntPtr.Zero, 1);if (_timerId1s == 0){Console.WriteLine("1s定时器创建失败");Cleanup();return;}Console.WriteLine("按下任意键退出...");Console.ReadKey();Cleanup();}private static void Cleanup(){if (_timerId1ms != 0){timeKillEvent(_timerId1ms);_timerId1ms = 0;}if (_timerId1s != 0){timeKillEvent(_timerId1s);_timerId1s = 0;}timeEndPeriod(1);}
}
运行后效果:
您将看到在控制台或Debug输出中,每秒打印一次计数值。与托管定时器相比,使用多媒体定时器后,每秒计数的值通常能更接近1000。尽管仍可能受到系统负载影响,但相较常规定时器,这已经是一个显著的提升。
总结
- 在C#中尝试实现1ms精度的定时器时,使用默认托管定时器往往不够精确。
- 系统定时器分辨率、托管运行时环境和线程调度机制都会影响定时精度。
- 通过使用
timeBeginPeriod
和多媒体定时器timeSetEvent
,我们可以大幅提升定时器的精度和稳定性。 - 若有更高的实时性需求,可考虑使用实时操作系统或硬件定时器来获得更高精度。
相关文章:
C#实现1ms定时器不精准?如何实现一个高性能高精度的1ms定时器?(附完整示例Demo)
在C#日常开发中,我们经常需要使用定时器(Timer)进行周期性任务的执行。 例如,每隔1秒打印一条日志,或每隔100毫秒执行某个数据刷新逻辑。 但是,当我们尝试在C#中实现一个1毫秒(1ms)…...
LeetCode 3. 无重复字符的最长子串
题目链接:3. 无重复字符的最长子串 首先想到的就是暴力破解,直接两层循环遍历,因为它说求无重复,那就可以用 set 来存储遍历到的字符,如果遍历到了同样的字符(在 set 中存在),就直接跳出第二层循环&#x…...
深度解析 Ansible:核心组件、配置、Playbook 全流程与 YAML 奥秘(上)
文章目录 一、ansible的主要组成部分二、安装三、相关文件四、ansible配置文件五、ansible 系列 一、ansible的主要组成部分 ansible playbook:任务剧本(任务集),编排定义ansible任务集的配置文件,由ansible顺序依次执…...
记一次由docker容器使得服务器cpu占满密码和密钥无法访问bug
Bug场景: 前几天在服务器上部署了一个免费影视网站,这个应用需要四个容器,同时之前的建站软件workpress也是使用docker部署的,也使用了三个容器。在使用workpress之前,我将影视软件的容器全部停止。 再使用workpress…...
功能篇:JAVA实现记住我功能
在Java Web应用程序中实现“记住我”功能,通常涉及以下几个步骤: 1. 创建一个持久化的标识符(如一个令牌或哈希值),并将其与用户账户关联。 2. 将这个标识符保存到客户端的cookie中。 3. 在服务器端,当用户…...
实现 DataGridView 下拉列表功能(C# WinForms)
本文介绍如何在 WinForms 中使用 DataGridViewComboBoxColumn 实现下拉列表功能,并通过事件响应来处理用户的选择。以下是实现步骤和示例代码。 1. 效果展示 该程序的主要功能是展示如何在 DataGridView 中插入下拉列表,并在选择某一项时触发事件。 2.…...
2024年上半年网络工程师综合知识真题及答案解析
2024年上半年网络工程师综合知识真题及答案解析 以下不属于5G网络优点的是()A.传输过程中消耗的资源少,对设备的电池更友好B.支持大规模物联网,能够连接大量低功耗设备,提供更高效的管理C.引入了网络切片技术,允许将物理网络划分为多个虚拟网络D.更好的安全性,采用更…...
数合平台功能-管理角色
前一阵,有朋友问到,看咱们产品的功能描述很强大,但很多功能看不到。为此,基于数据建模产品最新版本,和大家一起串一下产品的功能和使用路径。本节重点说一下管理角色有哪些功能 一、功能清单 从上图中可以看到&#x…...
LVGL9 开关控件 (lv_switch) 使用指南
文章目录 前言主体1. **控件概述**2. **控件的样式和组成部分**3. **使用控件**改变开关状态 4. **事件处理**5. **按键支持**6. **示例代码** 总结 前言 lv_switch 是 LittlevGL 提供的一个开关控件,外观类似一个小型滑块,常用于实现开关功能ÿ…...
麒麟 V10 系统(arm64/aarch64)离线安装 docker 和 docker-compose
前期准备 查看操作系统版本,跟本文标题核对一下 uname -a查看操作系统架构 uname -m下载离线包 下载 docker 离线包 地址:https://download.docker.com/linux/static/stable/ 选择系统架构对应的文件目录:aarch64,我目前使用…...
独立ip服务器有什么优点?
网站的性能和安全性直接影响到用户体验和业务发,独立IP服务器作为一种主流的托管方式,因其独特的优势而受到许多企业和个人站长的青睐。与共享IP相比,独立IP服务器到底有哪些优点呢? 使用独立IP的用户不必担心与其他网站共享同一…...
oracle之用户的相关操作
(1)创建用户(sys用户下操作) 简单创建用户如下: CREATE USER username IDENTIFIED BY password; 如果需要自定义更多的信息,如用户使用的表空间等,可以使用如下: CREATE USER mall IDENTIFIED BY 12345…...
深入浅出:PHP中的数组操作全解析
文章目录 引言理解数组创建数组使用方括号使用array()函数 访问数组元素数值索引数组关联数组 遍历数组使用for循环使用foreach循环 添加和修改数组元素添加元素修改元素 删除数组元素删除单个元素删除整个数组 多维数组创建多维数组访问多维数组元素 常用数组函数获取数组长度…...
2024年12月7日历史上的今天大事件早读
1732年12月07日英国皇家大剧院在伦敦开幕 1798年12月07日清代诗人袁枚逝世 1889年12月07日第一个充气轮胎受专利保护 1916年12月07日劳合-乔治出任英国首相 1926年12月07日第一台电冰箱受美国专利保护 1937年12月07日南京保卫战正式打响 1941年12月07日日本偷袭珍珠港 1…...
pymysql模块详解
华子目录 简介安装pymysql连接对象常用方法游标对象常用方法数据库操作查改批量增加删 使用with语句总结 简介 pymysql是一个用于Python编程的第三方模块,用于连接和操作MySQL数据库。它提供了一个简单而强大的接口,使开发者能够轻松地在Python程序中执…...
计算机网络研究实训室建设方案
一、概述 本方案旨在规划并实施一个先进的计算机网络研究实训室,旨在为学生提供一个深入学习、实践和研究网络技术的平台。实训室将集教学、实验、研究于一体,覆盖网络基础、网络架构、网络安全、网络管理等多个领域,以培养具备扎实理论基础…...
WEB安全 PHP学习
PHP基础 PHP编码显示问题 header ("Content-type: text/html; charsetgb2312"); header("Content-Type: text/html;charsetutf-8"); windows需要使用gbk编码显示 源码是 <?php header ("Content-type: text/html; charsetgb2312"); sys…...
Redis高阶之容错切换
当一台主机master宕掉之后,他的从机会取代主机么? 查看集群状态 127.0.0.1:6385> cluster nodes c8ff33e8da5fd8ef821c65974dda304d2e3327f9 192.168.58.129:638216382 slave f6b1fd5e58df90782f602b484c2011d52fc3482d 0 1733220836918 1 connecte…...
构建高效OTA旅游平台的技术指南
1. 引言 在信息技术高速发展的今天,互联网深刻地改变了人们的旅行方式。传统的旅行社模式逐渐被在线旅游平台所取代,OTA(Online Travel Agency,在线旅行社)旅游平台应运而生,成为人们获取旅游信息、预订旅…...
数据结构 (25)图的存储结构
前言 数据结构中的图是一种用于表示多对多关系的结构,其存储结构主要有两种:邻接矩阵和邻接表。 一、邻接矩阵 定义:邻接矩阵是一个二维数组,用于存储图中各个顶点之间的关系。数组的行和列分别代表图中的顶点,元素的值…...
【C语言】fscanf 和 fprintf函数
【C语言】fscanf 和 fprintf函数 文章目录 [TOC](文章目录) 前言一、定义二、代码例程三、实验结果四、参考文献总结 前言 使用工具: 1.编译器:DEVC 2.C Primer Plus 第六版-1 提示:以下是本篇文章正文内容,下面案例可供参考 一…...
C#调用c++创建的动态链接库dll文件
在C#中调用外部DLL文件是一种常见的编程实践,它具有以下几个重要意义:1.代码重用;2.模块化;3.性能优化;4.安全性;5.跨平台兼容性;6.方便更新和维护;7.利用特定技术或框架;…...
【数字电路与逻辑设计】实验一 序列检测器
文章总览:YuanDaiMa2048博客文章总览 【数字电路与逻辑设计】实验一 序列检测器 一、实验内容二、设计过程(一)作出状态图或状态表(二)状态化简(三)状态编码 三、源代码(一ÿ…...
沈阳工业大学《2024年827自动控制原理真题》 (完整版)
本文内容,全部选自自动化考研联盟的:《沈阳工业大学827自控考研资料》的真题篇。后续会持续更新更多学校,更多年份的真题,记得关注哦~ 目录 2024年真题 Part1:2024年完整版真题 2024年真题...
Javascript Clipper library, v6(介绍目录)
1.老祖宗C#版的Clipper2 Clipper2库可以对简单和复杂的多边形执行交集、并并、差分和异或布尔运算。它还执行多边形偏移 github地址:GitHub - AngusJohnson/Clipper2: Polygon Clipping and Offsetting - C, C# and Delphi 2.目前的移植版本 基于C#版的移植版本…...
uniapp+vue3+ts请求接口封装
1.安装luch-request yarn add luch-requestnpm install luch-request2.新建文件src/utils/request.ts 需要自己修改config.baseURL和token(获取存储的token) // import HttpRequest from luch-request; import type { HttpRequestConfig, HttpRespons…...
Spring Boot中的@GetMapping注解可以用于处理HTTP GET请求,并且可以接收对象参数,详细示例
下面内容来自Ai回答,经过亲自验证,正确 Spring Boot中的GetMapping注解可以用于处理HTTP GET请求,并且可以接收对象参数。 接收对象参数的基本方式 在Spring Boot中,可以通过GetMapping注解接收对象参数,这通…...
详解Vue设计模式
详解 vue 设计模式 Vue.js 作为一个流行的前端框架,拥有许多设计模式,这些设计模式帮助开发者更好地组织和管理代码,提升代码的可维护性、可扩展性和可读性。Vue 设计模式主要体现在以下几个方面: 1. 组件化设计模式 (Compon…...
webpack 题目
文章目录 webpack 中 chunkHash 和 contentHash 的区别loader和plugin的区别?webpack 处理 image 是用哪个 loader,限制 image 大小的是...;webpack 如何优化打包速度 webpack 中 chunkHash 和 contentHash 的区别 主要从四方面来讲一下区别&…...
Mysql - 存储引擎
一 MYSQL体系结构简介 MYSQL的体系结构可以分为四个层级,从上往下依次为: 1. 连接层: 最上层为客户端以及一些连接服务,包含连接操作,例如JAVA想要与MYSQL建立连接就需要用到JDBC,PHP语言与Python也可以连接到MYSQL&am…...
【实战教程】使用YOLOv8 OBB进行旋转框目标检测的数据集定义与训练【附源码】
《------往期经典推荐------》 一、AI应用软件开发实战专栏【链接】 项目名称项目名称1.【人脸识别与管理系统开发】2.【车牌识别与自动收费管理系统开发】3.【手势识别系统开发】4.【人脸面部活体检测系统开发】5.【图片风格快速迁移软件开发】6.【人脸表表情识别系统】7.【…...
怎么实现邮件营销自动化?
邮件营销能够出色地帮助我们与客户建立良好关系。无论是新客户还是老客户,都可以通过邮件来达成较为良好的客户关系。然而,从消费者的角度来看,每个人都有自己独特的习惯和特点,没有人希望收到千篇一律、营销意味过重的邮件。因此…...
华为服务器使用U盘重装系统
一、准备工作 下载官方系统(注意服务器CPU的架构是x86-64还是aarch64,不然可能报意想不到的错)制作启动U盘(下载rufus制作工具,注意文件系统选FAT32还是NTFS) 二、安装步骤 将U盘插入USB接口重启服务器…...
空安全编程的典范:Java 8中的安全应用指南
文章目录 一、Base64 编码解码1.1 基本的编码和解码1.2 URL 和文件名安全的编码解码器1.3 MIME Base64编码和解码 二、Optional类三、Nashorn JavaScript 一、Base64 编码解码 1.1 基本的编码和解码 Base64 编码: 使用 Base64.getEncoder().encodeToString(origin…...
深入解析 Loss 减少方式:mean和sum的区别及其在大语言模型中的应用 (中英双语)
深入解析 Loss 减少方式:mean 和 sum 的区别及其在大语言模型中的应用 在训练大语言模型(Large Language Models, LLM)时,损失函数(Loss Function)的处理方式对模型的性能和优化过程有显著影响。本文以 re…...
opencv4.8 ubuntu20.04源码编译 安装报错记录
-- IPPICV: Downloading ippicv_2021.8_lnx_intel64_20230330_general.tgz from https://raw.githubusercontent.com/opencv/opencv_3rdparty/1224f78da6684df04397ac0f40c961ed37f79ccb/ippicv/ippicv_2021.8_lnx_intel64_20230330_general.tgz make -j8 到这咋不动了 代理配…...
16-03、JVM系列之:内存与垃圾回收篇(三)
JVM系列之:内存与垃圾回收篇(三) ##本篇内容概述: 1、执行引擎 2、StringTable 3、垃圾回收一、执行引擎 ##一、执行引擎概述 如果想让一个java程序运行起来,执行引擎的任务就是将字节码指令解释/编译为对应平台上的本地机器指令才可以。 简…...
在 Spring Boot 中使用 JPA(Java Persistence API)进行数据库操作
步骤 1: 添加依赖 在 pom.xml 文件中添加相关依赖: <dependencies><!-- Spring Boot Starter Web --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><…...
【sqlserver】mssql 批量加载数据文件 bulk copy使用
参考文章: Using bulk copy with the JDBC driver SqlServer数据批量写入 SqlServer批量插入数据方法–SqlBulkCopy sqlserver buld copy需要提供,数据文件的对应表的元数据信息主要的字段的位置、字段的名称、字段的数据类型。 执行bulk load时候不一…...
卷积神经网络(CNN)的层次结构
卷积神经网络(CNN)是一种以其处理图像和视频数据的能力而闻名的深度学习模型,其基本结构通常包括以下几个层次,每个层次都有其特定的功能和作用: 1. 输入层(Input Layer): 卷积神经网…...
使用Excel的COUNTIFS和SUMIFS函数进行高级数据分析
使用Excel的COUNTIFS和SUMIFS函数进行高级数据分析 引言 在处理数据时,Excel 提供了多种内置函数来帮助用户快速获取所需信息。其中,COUNTIFS 和 SUMIFS 是两个非常强大的多条件聚合函数,它们允许你根据一个或多个标准来统计或汇总数据。本…...
上传ssh公钥到目标服务器
创建密钥 ssh-keygen -t rsa -b 4096 -C "xxxx.xx"上传 sudo ssh-copy-id -i /Users/xx/.ssh/id_rsa.pub root127.0.0.1...
在visio2021 中插入MathType公式
首先要确保有着两个软件,且能用。 1、打开visio2021,之后点击“插入”-“对象” 2、打开后,选择MathType,确定 3、确定后就会弹出MathType编辑器...
【计算机视觉】图像的几何变换
最常见的几何变换有仿射变换和单应性变换两种,最常用的仿射变换有缩放、翻转、旋转、平移。 1. 缩放 将图像放大或缩小会得到新的图像,但是多出的像素点如何实现----插值 1.1 插值方法 最近邻插值 双线性插值 cv2.resize() 是 OpenCV 中用于调整图像…...
IS-IS四
目录 点到点中LSP(类似LSA)的同步过程 注意LSP只有(1类LSA和2类LSA) 查看详细信息:display isis lsdb 0000.0000.0001.00-00 verbose 开摸: ISIS的伪节点LSP(类似LSA)没有路由信息 L1路由器的路由计算…...
CODA 离线安装及虚幻镜迁移
1、离线安装 1.1 下载Miniconda安装脚本 wget https://repo.anaconda.com/miniconda/Miniconda3-latest-Linux-x86_64.sh1.2 添加权限 chmod x Miniconda3-latest-Linux-x86_64.sh1.3 执行安装 ./Miniconda3-latest-Linux-x86_64.sh遇到问题,一路回车即可 1.4 …...
【Rive】混合动画
1 混合动画简介 【Rive】动画 中介绍了 Rive 中动画的基础概念和一般动画的制作流程,本文将介绍混合动画的基础概念和一般制作流程。Unity 中混合动画介绍详见→ 【Unity3D】动画混合。 混合动画是指同一时刻多个动画按照一定比例同时执行,这些动画控制的…...
软件体系结构复习-02 软件体系结构定位及构建
软件体系结构复习-02 软件体系结构定位及构建 原文链接:《软件体系结构复习-02 软件体系结构定位及构建》 目录 软件体系结构复习-02 软件体系结构定位及构建 1 什么是软件体系结构 2 软件生命周期中的软件体系结构 2.1 生命周期 2.2 定位与作用 1 规划和需求…...
MySQL-SQL语句
文章目录 一. SQL语句介绍二. SQL语句分类1. 数据定义语言:简称DDL(Data Definition Language)2. 数据操作语言:简称DML(Data Manipulation Language)3. 数据查询语言:简称DQL(Data Query Language)4. 数据控制语言:简称DCL(Data …...
Windows版Docker上不了网怎么办?
1、判断你的config文件、daemon文件的位置。 docker info命令输入, buildx: Docker Buildx (Docker Inc.) Version: v0.17.1-desktop.1 Path: C:\Users\AAA\.docker\cli-plugins\docker-buildx.exe 这个是你电脑这些文件的位置,修改linu…...