[原]C++新标准之std::chrono::duration
概览
c++新标准提供了新的线程库,最近在写测试代码的时候需要让当前线程休眠,之前直接调用windows提供的Sleep()就好了,新标准中可以使用std::this_thread::sleep_for()或者std::this_thread::sleep_until()
来实现休眠。其中涉及到了std::chrono::duration和std::chrono::time_point。本篇只总结std::chrono::duration,std::chrono::time_point会再写一篇总结。
std::chrono::duration
描述
std::chrono::duration定义在文件中,用来表示一个时间段。
cppreference上的原话如下:
Class template std::chrono::duration represents a time interval.
It consists of a count of ticks of type Rep and a tick period, where the tick period is a compile-time rational constant representing the number of seconds from one tick to the next.
The only data stored in a duration is a tick count of type Rep. If Rep is floating point, then the duration can represent fractions of ticks. Period is included as part of the duration's type, and is only used when converting between different durations.
Rep参数代表了可以传入的时间单位的类型,可以为float, int, int64等等,如果为float表示可以传入时间单位的一部分,比如传入1.2表示1.2倍个时间单位。
Period参数代表了时间单位,可以为微秒,毫秒,秒,分钟,小时等(或者其它自定义的单位,类型为std::ratio)。
注:
- 上文中的tick可以理解为周期,或时间单位。
- the number of seconds 表示是周期值基于秒来计算的。
类定义
std::chrono::duration是一个模板类,关键代码摘录如下(格式有调整):
- template<class _Rep, class _Period>
- class duration {
- public:
- typedef duration<_Rep, _Period> _Myt;
- typedef _Rep rep;
- typedef _Period period;
- // constructor, save param to _MyRep, used by count() member function.
- template<class _Rep2,
- class = typename enable_if<is_convertible<_Rep2, _Rep>::value
- && (treat_as_floating_point<_Rep>::value || !treat_as_floating_point<_Rep2>::value),
- void>::type>
- constexpr explicit duration(const _Rep2& _Val)
- : _MyRep(static_cast<_Rep>(_Val))
- {
- }
- constexpr _Rep count() const { return (_MyRep); }
- };
- // convert duration from one unit to another.
- template<class _To, class _Rep, class _Period> inline
- constexpr typename enable_if<_Is_duration<_To>::value, _To>::type
- duration_cast(const duration<_Rep, _Period>& _Dur)
- {
- typedef ratio_divide<_Period, typename _To::period> _CF;
- typedef typename _To::rep _ToRep;
- typedef typename common_type<_ToRep, _Rep, intmax_t>::type _CR;
- #pragma warning(push)
- #pragma warning(disable: 6326) // Potential comparison of a constant with another constant.
- return (_CF::num == 1 && _CF::den == 1
- ? static_cast<_To>(static_cast<_ToRep>(_Dur.count()))
- : _CF::num != 1 && _CF::den == 1
- ? static_cast<_To>(static_cast<_ToRep>(
- static_cast<_CR>(
- _Dur.count()) * static_cast<_CR>(_CF::num)))
- : _CF::num == 1 && _CF::den != 1
- ? static_cast<_To>(static_cast<_ToRep>(
- static_cast<_CR>(_Dur.count())
- / static_cast<_CR>(_CF::den)))
- : static_cast<_To>(static_cast<_ToRep>(
- static_cast<_CR>(_Dur.count()) * static_cast<_CR>(_CF::num)
- / static_cast<_CR>(_CF::den))));
- #pragma warning(pop)
- }
duration_cast()分析
函数duration_cast()提供了在不同的时间单位之间进行转换的功能。
duration_cast()主要分为两部分:
通过
ratio_divide定义了从一个ratio转换到另外一个ratio的转换比例。
比如1/10到2/5的转换比例是1/4((1/10/(2/5)) = 1/4),也就是说一个1/10相当于1/4个2/5。
对应到代码里就是_CF::num = 1, _CF::den = 4.根据转换比例把n个单位的原数据转换到目标数据(return语句)
return语句写的这么复杂是为了效率,避免不必要的乘除法,当分子是1的时候没必要乘,当分母是1的时候没必要除。
简化一下(去掉了强制类型转换)就是:
return _Dur.count() * (_CF::num / _CF::den);
通俗点讲:如果A到B的转换比例是num/den,那么1个A可以转换为num/den个B, n个A可以转换为 n * (num/den)个B。
注:vs自带的源码真心不易读,推荐参考boost源码。
预定义的duration
vs为了写代码方便,预定义了几个常用的时间单位,摘录如下:
- typedef duration<long long, nano> nanoseconds; // 纳秒
- typedef duration<long long, micro> microseconds; // 微秒
- typedef duration<long long, milli> milliseconds; // 毫秒
- typedef duration<long long> seconds; // 秒
- typedef duration<int, ratio<60> > minutes; // 分钟
- typedef duration<int, ratio<3600> > hours; // 小时
根据以上定义我们可以发现std::chrono::microseconds定义中的Rep的类型是long long,Period类型是milli。
注:因为
std::chrono::microseconds定义中的Rep的类型是long long, 我们不能通过如下方法来休眠100.5毫秒std::this_thread::sleep_for(std::chrono::microseconds(100.5));,类型不匹配,会报编译错误。如果想休眠100.5毫秒,我们可以这么写:
std::this_thread::sleep_for(std::chrono::duration<float, std::milli>(100.5f));
示例代码
例1:分钟转换为毫秒
- #include <iostream>
- #include <chrono>
- int main()
- {
- std::chrono::milliseconds ms = std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::minutes(3));
- std::cout << "3 minutes equals to " << ms.count() << " milliseconds\n";
- std::cin.get();
- }
例2. 自定义单位转换
- #include <iostream>
- #include <chrono>
- typedef std::chrono::duration<float, std::ratio<3, 1> > three_seconds;
- typedef std::chrono::duration<float, std::ratio<1, 10> > one_tenth_seconds;
- int main()
- {
- three_seconds s = std::chrono::duration_cast<three_seconds>(one_tenth_seconds(3));
- std::cout << "3 [1/10 seconds] equal to " << s.count() << " [3 seconds]\n";
- std::cin.get();
- }
例3. 休眠100毫秒
- #include <thread>
- #include <chrono>
- int main()
- {
- std::this_thread::sleep_for(std::chrono::milliseconds(100));
- // or
- std::this_thread::sleep_for(std::chrono::duration<long long, std::milli>(100));
- // or
- // typedef ratio<1, 1000> milli;
- std::this_thread::sleep_for(std::chrono::duration<long long, std::ratio<1, 1000> >(100));
- }
参考资料
- VS源码
- cppreference
[原]C++新标准之std::chrono::duration的更多相关文章
- [原]C++新标准之std::chrono::time_point
原 总结 STL 标准库 chrono time_point ratio 概览 类定义 总结 思考 拓展 system_clock steady_clock high_resolution_cloc ...
- [原]C++新标准之std::ratio
原 总结 ratio 概览 类定义 预定义ratio 应用 示例代码 参考资料 概览 std::ratio定义在<ratio>文件中,提供了编译期的比例计算功能.为std::chrono ...
- [原]C++新标准之std::thread
原 总结 C++11 thread 概览 std::thread 类定义 各个成员函数的简单介绍 例子 更多 参考资料 概览 从C++11开始提供了线程的支持,终于可以方便的编写跨平台的线程代码了. ...
- C++11 std::chrono库详解
所谓的详解只不过是参考www.cplusplus.com的说明整理了一下,因为没发现别人有详细讲解. chrono是一个time library, 源于boost,现在已经是C++标准.话说今年似乎又 ...
- c++11 时间类 std::chrono
概念: chrono库:主要包含了三种类型:时间间隔Duration.时钟Clocks和时间点Time point. Duration:表示一段时间间隔,用来记录时间长度,可以表示几秒钟.几分钟或者几 ...
- std::chrono计算程序运行时间
void CalRunTime() { auto t1=std::chrono::steady_clock::now(); //run code auto t2=std::chrono::steady ...
- C++11新特性,利用std::chrono精简传统获取系统时间的方法
一.传统的获取系统时间的方法 传统的C++获取时间的方法须要分平台来定义. 相信百度代码也不少. 我自己写了下,例如以下. const std::string getCurrentSystemTime ...
- C++ 11新特性:std::future & std::shared_future) (转载)
上一讲<C++11 并发指南四(<future> 详解二 std::packaged_task 介绍)>主要介绍了 <future> 头文件中的 std::pack ...
- c++11 标准库函数 std::move 和 完美转发 std::forward
c++11 标准库函数 std::move 和 完美转发 std::forward #define _CRT_SECURE_NO_WARNINGS #include <iostream> ...
随机推荐
- oracle数据库语言(2)----数据操作语言
DML(数据操纵语言)Data Manipulation Language 用于增.删.改 数据库中的数据 常伴随着TCL(事物操作语言)----commit(保存):rollback(回滚): ), ...
- Pycharm2020最新激活码|永久激活(附最新激活码和插件)
最近很多人的Pycharm激活时间又过期了,后台很多人索要激活码,我就再把激活的方法汇和工具再梳理一次给大家. 最主要有两种激活方式(两种方式需要的激活码不同): 一.激活码激活: 一般一年多需要激活 ...
- 实验吧web-难-认真一点!(布尔盲注,py脚本)
也可用bp进行爆破,这里用py脚本. 打看网页输入1,显示You are in,输入2,显示You are not in,是个布尔注入. 然后看看过滤了什么. sql注入没有过滤:--+.or sql ...
- SASS - 输出格式
SASS – 简介 SASS – 环境搭建 SASS – 使用Sass程序 SASS – 语法 SASS – 变量 SASS- 局部文件(Partial) SASS – 混合(Mixin) SASS ...
- 学习spring的第二天
对昨天的查漏:关于<bean>标签的scope属性,是由它决定原型和单例的,而不是说你java代码中用到了单例模式就是单例了. 其二就是lazy-init属性,它对于scope=" ...
- part5 城市页面列表开发
1.配置路由 先在router文件夹中,创建一个路由.引入组件 { path: '/city', name: 'HelloCity', component: city, meta: { name: ' ...
- IDEA--IDEA配置web项目
参考:https://blog.csdn.net/kfm1376822651/article/details/79666586 记学习springmvc时,使用idea部署web项目至tomcat. ...
- Uber推出全新交通估算体系能颠覆传统模式吗?
当下,大众的交通出行正在被全面颠覆.除了传统的出行方式外,共享打车.共享单车.共享滑板车.分时租赁的共享汽车等,正在形成一个全新交通出行矩阵.鉴于交通出行市场的巨大潜力,众多巨头及独角兽企业在绞尽脑汁 ...
- 题解 P1654 【OSU!】
题面 一序列\(a\), 对于每一个\(i\)均有\(a_i\)有\(p_i\)的几率为1, 否则为\(0\) 求: \(a\)中极长全\(1\)子序列长度三次方之和的期望 前置知识 基本期望(期望的 ...
- 使用那各VUE的打印功能(print.js)出现多打印一个空白页的问题
最近这段时间,用VUE写东西,有个打印功能. 百度了一下,铺天盖地的VUE打印的两种实现方法. 很感激这些千篇一律的帖子,虽然不知道他们是否真的用过,还是只是复制粘贴. 至少这些帖子告诉我,是有两个可 ...