当前位置: 首页 > 编程笔记 >

C++中Boost.Chrono时间库的使用方法

张勇
2023-03-14
本文向大家介绍C++中Boost.Chrono时间库的使用方法,包括了C++中Boost.Chrono时间库的使用方法的使用技巧和注意事项,需要的朋友参考一下

前言

大家应该都有所体会,时钟这个东西在程序中扮演者重要的角色,在系统编程的时候睡眠、带超时的等待、带超时的条件变量、带超时的锁都会用到,但是往往对特定系统依赖性很大,感觉即使不考虑系统的跨平台性,如果能使用一个稳定的接口,同时如果能够方便的对时刻、时段等进行相关的操作和运算,将是再好不过的了。

在boost库中和时间相关的库有Boost.DateTime和Boost.Chrono,前者专注于时间时刻以及本地化相关的内容,而后者主要是时刻、时长和时间的计算等内容。当然,C++11标准已经支持std::chrono了,但是为了兼容老编译系统现在很多C++库和程序都使用boost.chrono作为时间类库(还有的原因就是std::chrono没有收录boost.chrono的所有功能,比如统计CPU使用时间、自定义时间输出格式等),不过比较可惜的是即便使用boost::chrono作为权宜之计,也需要boost-1.47版本之上才行,而现在比较旧的发行版需要升级boost库才可以使用。想想现在RHEL-6.x仍然被大规模的部署,而且RedHat要为这货提供长达十年的技术支持,真不知道啥时候才能顺顺利利的享受C++11……

Boost.Chrono的时间类型分为duration和time_point,也就是时长和时刻两类,很多概念和接口都是围绕这两个维度去定义和实现的。

一、Clock

clock是Boost.Chrono中的重要概念,而且这些clock都包含一个now()的成员函数,用于返回当前的time_point。Boost.Chrono包含的clock类型有:

  (1) chrono::system_clock 代表系统时间,比如电脑上显示的当前时间,其特点是这个时间可以被用户手动设置更新,所以这个时钟是可以和外部时钟源同步的。这个时钟还有一个to_time_t()成员函数,用于返回自1970.1.1开始到某个时间点所经过的秒数,数据类型是std::time_t。这种时钟通常用来转换成日历时间使用。

  (2) chrono::steady_clock 其特点是时间是单调增长的,后一个时刻访问得到的时间点肯定比之前时刻得到的时间点要晚,即使我们手动将系统时间向前调整了也不会改变这个时钟稳步向前推行累计,其也被称为monotonic time,该时钟是均匀增长且不能被调整,其特性对于很多不允许时间错乱的系统是十分重要的。chrono::steady_clock通常是基于系统启动时间来计时的,而且常常用来进行耗时、等待等工作使用。

  (3) chrono::high_resolution_clock 依赖于系统实现,通常是上面两种时钟的某个宏定义,取决于哪个时钟源更为的精确,所以其输出也决定于取决于上面哪个clock来实现的。

  (4) chrono::process_real_cpu_clock 表示自进程启动以来使用的CPU时间,而这个数据也可以通过使用std::clock()来获得。chrono::process_user_cpu_clock、boost::chrono::process_system_cpu_clock表示自进程启动以来,在用户态、内核态所花费的时间,而所有的这些事件可以通过chrono::process_cpu_clock来获得,他返回上面所有时间组成的一个tuple结构。

  (5) chrono::thread_clock 返回基于线程统计的花费时间,而且不区分用户态、内核态的时间。

二、time_point

time_point代表时间点,其等价于某个时刻(clock)+duration的结果,同时两个time_point做减法也可以得到一个duration。time_point常见的描述为:3分钟之后、2038年1月1日10:32:23、定时器启动后的20ms……

上面的clock都有一个now()成员函数,其返回的就是chrono::time_point类型。这个类型使用一个模板来实现的,所以其实际类型极度依赖于所选择的时钟源。

Boost.Chrono有一个chrono::time_point_cast转换函数,可以显式从高粒度向低粒度对time_point进行转换。

chrono::process_real_cpu_clock::time_point p = chrono::process_real_cpu_clock::now();
std::cout << chrono::time_point_cast<minutes>(p) << '\n';

三、duration

关于时长,chrono::duration也是一个模板类型,其第一个模板参数表明存储所用的数据类型(int、long、double等),第二个模板参数表示ratio(比如24、60、1000等)。为了方便用户的使用,Boost.Chrono提供了duration常用的六种时间类型,且他们都用一个足够大的整数进行内部保存,其计量值可以用count()成员函数得到:

chrono::nanoseconds、chrono::milliseconds、chrono::microseconds、chrono::seconds、chrono::minutes、chrono::hours,而且为这些类型都重载了计算操作符:+、-、<等,方便时间的计算和比较。文档说传统的Boost.DateTime是用继承实现的,相比而言Boost.Chrono更加的简洁高效,而且和前者保持了一致的接口。

chrono::steady_clock::time_point start = chrono::steady_clock::now();
...
chrono::duration<double> sec = chrono::steady_clock::now() - start;
std::cout << "we took " << sec.count() << " seconds\n";
auto go = chrono::steady_clock::now() + chrono::nanoseconds(500); 
while (chrono::steady_clock::now() < go) 
... ;

上面的六种time_point类型表示的维度不一,粗粒度的时长肯定能用细粒度的类型表示,反之则可能丢失精度,所以需要使用chrono::duration_cast()函数做显式的转换。

可能上面六种类型的时间不咋的,但重点是现在boost::chrono被广为使用在boost的其他库里面,比如我们看一个条件变量的带超时等待的原型:

template< typename Clock, typename Duration >
cv_status wait_until( std::unique_lock< mutex > & lk,
      std::chrono::time_point< Clock, Duration > const& abs_time);
      
template< typename Rep, typename Period >
cv_status wait_for( std::unique_lock< mutex > & lk,
     std::chrono::duration< Rep, Period > const& rel_time);

这就意味着我们可以直接将chrono::minutes{2}这样的duration对象丢给这个函数就好了,创建任意精度的时长都很方便,而不用像以前一样关注函数接口有人用seconds、有人用milliseconds、有人用timeval了。

现在boost库和标准库中,基于时间段超时的函数都具有for后缀,而基于时间点超时的变量具有until后缀,比如this_thread::sleep_for()和this_thread::sleep_until()类似的还有:wait、try_lock、unique_lock用于条件变量、mutex互斥、unique_lock操作。

四、自定义格式的时间输出

通过time_fmt()可以对时刻进行格式化输出,使用的时候需要包含头文件。

time_fmt(boost::chrono::timezone::local, "%H:%M:%S");
time_fmt(boost::chrono::timezone::utc, "%H:%M:%S");

参考

Chapter 37. Boost.Chrono

Chapter 8. Boost.Chrono 2.0.5

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对小牛知识库的支持。

 类似资料:
  • 本文向大家介绍C#中的Char.IsNumber()方法使用,包括了C#中的Char.IsNumber()方法使用的使用技巧和注意事项,需要的朋友参考一下 c#中的Char.IsNumber()方法用于表示指定的Unicode字符是否被归类为数字。 语法 下面是语法 在上面,参数ch是要计算的Unicode字符。 示例 现在让我们看一个实现Char.IsNumber()方法的示例 输出结果 示例

  • 本文向大家介绍Java时间类库Timer的使用方法与实例详解,包括了Java时间类库Timer的使用方法与实例详解的使用技巧和注意事项,需要的朋友参考一下 使用 Java 来调度定时任务时,我们经常会使用 Timer 类搞定。Timer 简单易用,在一些业务场景下用来实现简单定时调度。 Jave时间类库Timer简单使用 创建Timer对象 编写 自己的 task 类,该类集成 TimerTask

  • 本文向大家介绍C语言中时间戳转换成时间字符串的方法,包括了C语言中时间戳转换成时间字符串的方法的使用技巧和注意事项,需要的朋友参考一下 在PE格式里有个字段是文件的创建时间戳,我想把转成字符串,这样看的更直观。 以上所述是小编给大家介绍的C语言中时间戳转换成时间字符串的方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对呐喊教程网站的支持!

  • 本文向大家介绍C++取得当前时间的方法,包括了C++取得当前时间的方法的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了C++取得当前时间的方法,分享给大家供大家参考。 具体实现方法如下: 希望本文所述对大家的C++程序设计有所帮助。

  • 这是我的代码:- 我已经仔细检查了所有的身份验证参数。 将主文件夹中的文件保存为状态。py后运行python状态。py出现以下错误:- 请帮帮我

  • 本文向大家介绍lua中使用毫秒精度时间的方法,包括了lua中使用毫秒精度时间的方法的使用技巧和注意事项,需要的朋友参考一下 lua自带的时间函数只能到秒的精度。 为了统计到毫秒精度的时间,可以使用luasocket。下载地址http://files.luaforge.net/releases/luasocket/luasocket 编译安装的时候,你可能需要在源码包根目录下的config文件中指定