C++11之前,想要获取时间并对其打印是有些困难的,因为C++并没有标准时间库。想要对时间进行统计就需要调用C库,并且我们要考虑这样的调用是否能很好的封装到我们的类中。
C++11之后,STL提供了 chrono
库,其让对时间的操作更加简单。本节,我们将会使用本地时间,并对本地时间进行打印,还会给时间加上不同的偏移,这些操作很容易使用 std::chrono
完成。
#include <iostream>
#include <iomanip>
#include <chrono>
using namespace std;
using namespace std::literals;
ostream &operator<<(ostream &os,
const chrono::time_point<chrono::system_clock>
&t)
{
const auto tt(chrono::system_clock::to_time_t(t));
const auto loct(std::localtime(&tt));
return os << put_time(loct, "%c");
}
int main()
{
auto now(chrono::system_clock::now());
cout << "现在时间为 " << now << '\n';
chrono::hours chrono_12h{12};
cout << "加上十二小时后:"
<< (now + chrono_12h) << '\n';
cout << "减去 12 小时 15分钟后:"
<< (now - 12h - 15min) << '\n';
}
现在时间为 03/04/23 10:58:28
加上十二小时后:03/04/23 22:58:28
减去 12 小时 15分钟后:03/03/23 22:43:28
我们可以通过 std::chrono::system_clock
来获取当前时间点。这个STL时钟类是唯一一个能将时间点的值转换成一个时间结构体的类型,其能将时间点以能够看懂的方式进行输出。
为了打印这样的时间点,我们可以对 operator<<
操作符进行重载:
ostream &operator<<(ostream &os,
const chrono::time_point<chrono::system_clock>
&t)
{
const auto tt(chrono::system_clock::to_time_t(t));
const auto loct(std::localtime(&tt));
return os << put_time(loct, "%c");
}
首先,将 chrono::time_point<chrono::system_clock>
转换为 std::time_t 。然后,使用 std::localtime
将这个时间值进行转换,这样就能获取到一个本地时钟的相对时间值。这个函数会给我们返回一个转换后的指针(对于这个指针背后的内存不用我们多操心,因为其是一个静态对象,并不是从堆上分配的内存),这样我们就能完成最终的打印。
std::put_time
函数接受一个流对象和一个时间格式字符串。 %c
表示标准时间格式字符串,例如 Sun Mar 12 11:33:40 2017
。我们也可以写成 %m/%d/%y
;之后,时间就会按照这个格式进行打印,比如 03/12/17
。时间格式符的描述很长,想要了解其具体描述的最好方式就是去查看C++参考手册。
除了打印,我们也会为我们的时间点添加偏移。这也很简单,比如:12小时15分钟就可以表示为 12h+15min
。 chrono_literals
命名空间为我们提供了字面类型: hours(h), minutes(min), seconds(s), milliseconds(ms), microseconds(us),nanoseconds(ns)
。通过对两个时间间隔的相加,我们会得到一个新的时间点。