0

0

c++如何实现一个定时器 c++高精度Timer实现【实例】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-02 15:06:08

|

709人浏览过

|

来源于php中文网

原创

高精度定时器需用steady_clock+独立线程实现,避免sleep_for低精度问题;支持单次/周期回调,结合自旋与sleep提升亚毫秒级精度。

c++如何实现一个定时器 c++高精度timer实现【实例】

在 C++ 中实现高精度定时器,核心是利用操作系统提供的高分辨率时间源和异步/轮询机制,避免依赖 std::this_thread::sleep_for 这类低精度、易受系统调度影响的方式。下面给出一个跨平台(Windows/Linux)友好的、基于 C++11 标准的高精度 Timer 实例,支持毫秒级甚至亚毫秒级精度(取决于系统支持),并可回调函数或 lambda。

使用 std::chrono + std::thread 实现轻量级高精度 Timer

这是最常用、无需第三方库、可移植性强的方案。关键点在于:用 std::chrono::steady_clock(单调时钟,不受系统时间调整影响)做计时基准,用独立线程执行延时逻辑,避免阻塞主线程。

示例:一个可重复触发的高精度单次/周期 Timer 类:

#include <iostream>
#include <chrono>
#include <thread>
#include <atomic>
#include <functional>
<p>class HiResTimer {
std::atomic<bool> running<em>{false};
std::thread t</em>;
std::function<void()> callback_;
std::chrono::steady<em>clock::duration interval</em>;
bool is<em>periodic</em>;</p><p>public:
explicit HiResTimer(std::function<void()> cb, 
std::chrono::steady<em>clock::duration interval,
bool periodic = false)
: callback</em>(std::move(cb)), interval_(interval), is<em>periodic</em>(periodic) {}</p><pre class='brush:php;toolbar:false;'>void start() {
    if (running_.exchange(true)) return;
    t_ = std::thread([this] {
        auto next = std::chrono::steady_clock::now() + interval_;
        while (running_) {
            auto now = std::chrono::steady_clock::now();
            if (now >= next) {
                callback_();
                if (is_periodic_) {
                    next += interval_;
                } else {
                    running_ = false;
                    break;
                }
            } else {
                // 高精度休眠:自旋 + sleep 组合减少唤醒延迟
                auto sleep_dur = next - now;
                if (sleep_dur > std::chrono::microseconds(100)) {
                    std::this_thread::sleep_for(sleep_dur - std::chrono::microseconds(50));
                }
                // 剩余微小时间用空循环微调(慎用,仅对 sub-ms 场景且 CPU 允许时)
                while (std::chrono::steady_clock::now() < next && running_);
            }
        }
    });
}

void stop() {
    running_ = false;
    if (t_.joinable()) t_.join();
}

};

立即学习C++免费学习笔记(深入)”;

使用方式:

Veo
Veo

Google 最新发布的 AI 视频生成模型

下载
int main() {
    int count = 0;
    HiResTimer timer([&count]() {
        std::cout << "Tick #" << ++count << " at " 
                  << std::chrono::duration_cast<std::chrono::microseconds>(
                         std::chrono::steady_clock::now().time_since_epoch()).count()
                  << " μs\n";
    }, std::chrono::milliseconds(10), true); // 每 10ms 触发一次
<pre class='brush:php;toolbar:false;'>timer.start();
std::this_thread::sleep_for(std::chrono::seconds(1));
timer.stop();

}

Windows 平台:使用 QueryPerformanceCounter 实现纳秒级精度

若只面向 Windows,QueryPerformanceCounter(QPC)提供硬件级高精度计时(通常可达 ~100ns 级),比 steady_clock 更稳定(尤其在旧系统上)。需配合 QueryPerformanceFrequency 换算。

  • 适合对抖动敏感的场景(如音视频同步、实时采集)
  • 注意:QPC 在某些虚拟机或老旧 BIOS 上可能不可靠,建议运行时检测有效性
  • 不推荐直接裸用 Sleep,应结合 waitable timer 或自旋+sleep 混合策略

Linux 平台:clock_gettime(CLOCK_MONOTONIC_HR) + timerfd_create

Linux 下更“正统”的高精度定时方案是:

  • clock_gettime(CLOCK_MONOTONIC_HR) 提供纳秒级单调时钟(glibc 2.17+)
  • timerfd_create 创建文件描述符式定时器,可与 epoll 集成,无忙等、无信号、线程安全
  • 适合服务端长时间运行、需响应 I/O 事件的定时任务(如超时管理、心跳)

简单示意(省略错误处理):

#include <sys/timerfd.h>
#include <unistd.h>
#include <poll.h>
<p>int tfd = timerfd_create(CLOCK_MONOTONIC, 0);
struct itimerspec ts = {};
ts.it_value.tv_sec = 0;
ts.it_value.tv_nsec = 10'000'000; // 10ms
ts.it_interval = ts.it_value;
timerfd_settime(tfd, 0, &ts, nullptr);</p><p>// 后续可用 read(tfd, &exp, sizeof(exp)) 获取超时次数,或用 poll 等待</p>

进阶建议:避免常见精度陷阱

  • 别用 std::chrono::system_clock 做定时——它可能被 NTP 调整,导致跳变
  • 避免纯 busy-wait(while loop)耗尽 CPU;合理搭配 sleep + 精确校准
  • 定时器回调中避免长耗时操作,否则会累积误差;必要时用队列+工作线程解耦
  • 多核环境下注意缓存一致性:std::atomic_thread_fencememory_order_relaxed 的合理使用
  • 实测精度前,先用 perf stat -e task-clock,cycles,instructions(Linux)或 VTune(Windows)分析调度抖动

不复杂但容易忽略:真正决定“高精度”的不只是 API,而是你的线程优先级、CPU 亲和性设置、系统负载以及是否禁用节能模式(如 Intel SpeedStep / Linux cpupower governor)。生产环境建议搭配 pthread_setschedparam(SCHED_FIFO)或 Windows 的 SetThreadPriority 提升实时性。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

104

2023.09.25

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

214

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

192

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

60

2026.01.05

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

743

2023.08.10

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

743

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

97

2025.12.01

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

1386

2023.07.26

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

2

2026.03.05

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 10.1万人学习

Git 教程
Git 教程

共21课时 | 4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号