C++11中的chrono库
创始人
2025-05-28 22:33:52
0

文章目录

  • 前言
  • 1、记录时长的duration
  • 2、表示时间点的time_point
  • 3、获取系统时钟的clocks

前言

C++11提供了日期时间相关的库chrono,通过chrono库可以很方便的处理日期和时间。chrono库主要包含3种类型:时间间隔duration、时钟clocks和时间点time_point

1、记录时长的duration

duration为一个模板类,表示时间间隔,可以是几秒、几分钟或者几个小时的时间间隔。duration的原型如下:

template  >
class duration;

第一模板参数Rep是一个数值类型,表示时钟数的类型;第二个模板参数是一个默认模板参数std::ratio,表示时钟周期,它的原型如下:

template  class ratio;

它表示每个时钟周期的秒数,其中第一个模板参数N代表分子,D代表分母,分母默认为1,因此,ratio代表的是一个分子除以分母的分数值。比如:

ratio<2>	//代表2秒
ratio<60>	//代表1分钟
ratio<60*60>	//代表1小时
ratio<60*60*24>	//代表1天
ratio<1, 1000>	//代表1毫秒
ratio<1, 1000000>	//代表1微秒
ratio<1, 1000000000>	//代表1纳秒

为了方便使用,标准库定义了一些常用的时间间隔,如时、分、秒、毫秒、微秒和纳秒,在chrono命名空间下,定义如下(vs2013的源码):

typedef duration nanoseconds;
typedef duration microseconds;
typedef duration milliseconds;
typedef duration seconds;
typedef duration > minutes;
typedef duration > hours;

通过定义这些常用的时间间隔类型,我们能方便的使用它们,比如线程休眠:

//休眠100毫秒
this_thread::sleep_for(std::chrono::duration>(100));
this_thread::sleep_for(std::chrono::microseconds(100));//更简单//休眠3秒
this_thread::sleep_for(std::chrono::duration(3));
this_thread::sleep_for(std::chrono::seconds(3));//更简单

chrono还提供了获取时间间隔的时钟周期数的方法count(),它的基本用法如下:

#include
#include
int main()
{std::chrono::seconds s(3);//3秒std::chrono::milliseconds ms = 2 * s;//6000毫秒std::cout << "3 s duration has " << s.count() << " ticks\n" << "6000 ms duration has " << ms.count() << " ticks\n";
}

执行结果:

在这里插入图片描述

duration在某些情况下可以进行转换,例如,当duration的Rep都为整型,且源Period要大于目标Period时或者目标duration的Rep为浮点数时可以使用传统类型转化或者隐式调用其单值构造函数,不必调用duration_cast

int main()
{//目标duration的Rep为doublestd::chrono::milliseconds int_milliseconds(1024);       // 1024msstd::chrono::microseconds int_microseconds(1024);       // 1024usstd::chrono::duration double_seconds;double_seconds = int_microseconds;	//1024ms = 1.024sdouble_seconds = int_milliseconds;	//1024us = 0.001024s//当duration的Rep都为整型,且源Period可被目标Period整除时int_microseconds = int_milliseconds; //ms赋值给us可以,但是us赋值给ms不可int_milliseconds = std::chrono::seconds(1);	//s赋值给ms可以,但是ms赋值给s不可以//源duration的Rep为double, 目标duration的Rep不为double,不能转换//std::chrono::milliseconds t1 = std::chrono::duration(1024);/** 数据会发生截断时的转化* chrono库提供了duration之间相互转化的函数,其定义如下*      template *      constexpr ToDuration duration_cast(const duration& d);
*/std::chrono::seconds t1 = std::chrono::duration_cast(std::chrono::milliseconds(1024));               // 1s = 1024ms(精度损失)std::chrono::seconds t2 = std::chrono::duration_cast(std::chrono::duration(1.024));          // 1s = 1.024s 
}

时间间隔之间可以做运算,计算两端时间间隔的差值的实例如下:

int main()
{std::chrono::minutes t1(10);std::chrono::seconds t2(50);std::chrono::seconds t3 = t1 - t2;cout << t3.count() << " second" << endl;
}

执行结果:

在这里插入图片描述

其中,t1代表10分钟,t2代表50秒,t3则是t1减t2,也就是600-50=550秒。通过调用t3的count()输出差值550个时钟周期,因为t3的时钟周期为1秒,所以t3表示时间间隔为550秒。

值得注意的是,duration的加减运算有一定的规则,当两个duration时钟周期不相同的时候,会先统一成一种时钟,然后再作加减运算。统一成一种时钟的规则如下:

对于ratiocount1和ratiocount2。如果x1、x2的最大公约数为x,y1、y2的最小公倍数为y,那么统一之后的ratio为ratio

例如:

int main()
{std::chrono::duration> d1(3);std::chrono::duration> d2(1);auto d3 = d1 - d2;cout << "d3类型 : "<

执行结果:

在这里插入图片描述

根据前面介绍的规则,对于9/7和6/5,分子取最大公约数3,分母取最小公倍数35,所以,统一之后的duration为std::chrono::duration>。然后再将原来的duration转换为统一的duration,最后计算的时钟周期数为:((9/7)/(3/35)*3)-((6/5)/(3/35)*1),结果为31

2、表示时间点的time_point

time_point表示一个时间点,用来获取从它的clock的开始所经过的duration(比如,可能是1970.1.1以来的时间间隔)和当前时间,可以做一些时间的比较和算术运算,可以喝ctime库结合起来显示时间。time_point必须用clock来计时。time_point有一个函数time_from_eproch()用来获得1970年1月1日到time_point时间经过的duration

time_point是一个类模板,原型如下:

template 
class time_point;

第一个模板参数Clock用来指定所要使用的时钟(标准库中有三种时钟,system_clock,steady_clock和high_resolution_clock)
第二个模板函数参数用来表示时间的计量单位(特化的std::chrono::duration<> )

计算当前时间距离1970年1月1日有多少天

#include
#include
#include
using namespace std::chrono;
int main()
{using days_type = duration>;time_point today = time_point_cast(system_clock::now());std::cout << today.time_since_epoch().count() << " days since epoch" << endl;
}

执行结果:

在这里插入图片描述

time_point还支持一些算术运算,比如两个time_point的差值时钟周期数,还可以和duration相加减。要注意不同clock的time_point是不能进行算术运算的。下面例子将展示前一天和后一天的日期:

#include
#include
#include
using namespace std::chrono;
int main()
{system_clock::time_point now = system_clock::now();std::time_t prev = system_clock::to_time_t(now - hours(24)); //返回时间戳std::time_t next = system_clock::to_time_t(now + hours(24)); //返回时间戳cout << "One day ago, the time was " << std::put_time(std::localtime(&prev), "%Y.%m.%d %H:%M:%S") << endl;cout << "A day later, the time is " << std::put_time(std::localtime(&next), "%Y.%m.%d %H:%M:%S") << endl;	 
}

执行结果:

在这里插入图片描述

3、获取系统时钟的clocks

clocks表示当前的系统时钟,内部有time_point、duration、Rep、Period等信息,主要用来获取当前时间,以及实现time_t和time_point的相互转换。clocks包含如下3种时钟:
-system_clock:代表真实时间的挂钟时间,具体时间依赖于系统。system_clock保证提供的时间值是一个可读时间
-steady_clock:不能被“调整” 的时钟,并不一定代表真实世界的挂钟时间。保证先后调用now()得到的时间值是不会递减的
-high_resoulution_clock:高精度时钟,实际上是system_clock或者steady_clock的别名。可以通过now()来获取当前时间点,代码如下:

int main()
{std::chrono::system_clock::time_point t1 = std::chrono::system_clock::now();cout << "hello fl" << endl;std::chrono::system_clock::time_point t2 = std::chrono::system_clock::now();cout << (t2 - t1).count() << " tick count" << endl;return 0;
}

执行结果:

在这里插入图片描述

system_clock的to_time_t方法可以将一个time_point转换为ctime:

std::time_t now_c = std::chrono::system_clock::to_time_t(time_point);

而from_time_t方法则正好相反,它将ctime转换为time_point

steady_clock可以获取稳定可靠的时间间隔,后一次调用now()的值和前一次的差值不会因为修改了系统时间而改变,从而保证了稳定的时间间隔。steady_clock的用法和system用法一样。

相关内容

热门资讯

银价推涨光伏组件报价,下游企业... 来源:第一财经 受成本端银价上涨影响,本周光伏组件价格再次上调。据行业机构Infolink Cons...
黄金史诗级暴跌,原因可能与一纸... 当地时间1月30日,随着美联储前理事凯文·沃什(Kevin Warsh)正式被美国总统特朗普提名为下...
深圳国资七亿下场扫货白石洲? 来源:市场资讯 (来源:深圳房产在线) 最近看到,近日一则消息引发关注,就是今年1月发生一宗白石洲大...
国投智能2025业绩承压 AI... 来源:财联社 财联社1月30日讯(记者 方彦博)2025年,AI应用的商业化落地是众多AI企业面临的...
原创 男... 在爱情的海洋中,星座的波涛有时能揭示出隐藏的情感暗流。当男人在愤怒的风暴中显露出四种迹象时,或许他并...
农业银行董事长谷澍会见英格兰银... 来源:市场资讯 来源:中国农业银行 1月29日,农业银行董事长谷澍会见了英格兰银行副行长兼英国审慎监...
“易中天”,业绩大爆发!需求增... “易中天”2025年度业绩持续爆发! 1月30日晚间,中际旭创发布2025年度业绩预告,预计2025...
双平台战略提速:仙乐健康谋“A... 中国营养健康食品行业的龙头企业仙乐健康,在1月30日向市场投下了一枚重磅消息:公司已正式向香港联交所...
左季庆染指淳厚基金股权纷争为谁... 2026年1月6日,证监会一纸批复核准上海长宁国有资产经营投资有限公司(下称“长宁国资”)成为淳厚基...
上市即巅峰?拉芳家化首度亏损,... 为什么消费端对“拉芳”爱不起来了? 作者 | 方璐 编辑丨于婞 来源 | 野马财经 拉芳家化(603...
原创 黄... 1月31日晚间,英伟达CEO黄仁勋现身中国台湾台北市砖窑古早味怀旧餐厅,宴请了35位与英伟达合作的供...
山西太钢不锈钢股份有限公司 2... 来源:证券日报 证券代码:000825 证券简称:太钢不锈 公告编号:2026-001 本公司及董...
把自己的银行贷款出借给别人,有... 新京报讯(记者张静姝 通讯员邸越洋)因贷款出借后未被归还,原告牛女士将被告杨甲、杨乙诉至法院,要求二...
金价暴跌,刚买的金饰能退吗?有... 黄金价格大跌,多品牌设置退货手续费。 在过去两三天,现货黄金价格经历了“过山车”般的行情,受金价下跌...
预计赚超2500万!“豆腐大王... 图片来源:图虫创意 在经历了一年亏损后,“豆腐大王”祖名股份(003030.SZ)成功实现扭亏为盈。...
特朗普提名“自己人”沃什执掌美... 据新华社报道,当地时间1月30日,美国总统特朗普通过社交媒体宣布,提名美国联邦储备委员会前理事凯文·...
爱芯元智将上市:连年大额亏损,... 撰稿|多客 来源|贝多商业&贝多财经 1月30日,爱芯元智半导体股份有限公司(下称“爱芯元智”,HK...
一夜之间,10只A股拉响警报:... 【导读】深康佳A等10家公司昨夜拉响退市警报 中国基金报记者 夏天 1月30日晚间,A股市场迎来一波...
谁在操控淳厚基金?左季庆为谁趟... 2026年1月6日,证监会一纸批复核准上海长宁国有资产经营投资有限公司(下称“长宁国资”)成为淳厚基...
工商银行党委副书记、行长刘珺会... 人民财讯1月31日电,1月29日,工商银行党委副书记、行长刘珺会见来访的上海电气集团党委书记、董事长...