我做了两个函数来计算当天的开始时间戳(即当天的00:00:00)和给定纪元时间戳的小时(从1到24开始)。
#include <cstdint>
#include <ctime>
const uint8_t FIRST_HOUR = 0x01; // 01, 02, ..., 24
const uint32_t SECS_PER_HOUR = 3600; // 3600 secs per hour
uint32_t CalcDaiBaseTimestamp(uint32_t in_ts) {
time_t ts = in_ts;
struct tm timeinfo = *localtime(&ts);
timeinfo.tm_hour = 0;
timeinfo.tm_min = 0;
timeinfo.tm_sec = 0;
time_t tmp_base_ts = mktime(&timeinfo);
return (uint32_t)tmp_base_ts;
}
void CalcDaiBaseTimestampAndHour(uint32_t in_ts,
uint32_t& base_ts,
uint8_t& hour_nth) {
base_ts = CalcDaiBaseTimestamp(in_ts);
hour_nth = (in_ts - base_ts) / SECS_PER_HOUR + FIRST_HOUR;
}
CalcDaiBaseTimestampAndHour
是从多个线程调用的。
代码使用g++ (Ubuntu 4.8.4-2ubuntu1~14.04.4) 4.8.4
编译,程序在Ubuntu 14.04 x64
上运行。
大部分时间我的程序运行良好,但我有时会观察到一些“怪异”的结果,如下所示:
(timestamp: 1554459477.500) -> (base: 1553990400, hour_nth: 131)
虽然正确的结果应该是:
(timestamp: 1554459477.500) -> (base: 1554422400 / hour_nth: 11)
因为:
1554459477.500 = 2019-04-05 10:17:57.500
base_ts = 2019-04-05 00:00:00 = 1554422400
hour_nth = 11
由于问题有时会发生,所以我认为原因可能是某些ctime
相关函数的线程安全性。
什么可能导致“怪异”的结果?请帮我解决这个问题!如果原因实际上是与ctime
相关的函数的线程安全性,那么我该如何解决这个问题(例如某些C ++ 11标准库)?
你能告诉我如何使用日期库解决这个问题吗?
参考链接:https://github.com/HowardHinnant/date
码:
#include "date/date.h"
#include <iomanip>
#include <iostream>
int
main()
{
using namespace std::chrono;
using namespace date;
using dsec = duration<double>;
sys_time<dsec> tp{dsec{1554459477.500}};
std::cout << std::setprecision(3) << std::fixed
<< tp.time_since_epoch().count()
<< " = " << round<milliseconds>(tp) << '\n';
sys_seconds base_ts = floor<days>(tp);
std::cout << "base_ts = " << base_ts << " = "
<< base_ts.time_since_epoch().count() << '\n';
auto hour_nth = floor<hours>(tp - base_ts) + hours{1};
std::cout << "hour_nth = " << hour_nth.count() << '\n';
}
输出:
1554459477.500 = 2019-04-05 10:17:57.500
base_ts = 2019-04-05 00:00:00 = 1554422400
hour_nth = 11
笔记: