获得绝对时间的更好方法?
Better way to get absolute time?
目前我正在尝试获取绝对时间以用于 pthread_mutex_timedlock
。我知道我需要将 timeval
从 gettimeofday
添加到 timespec
,然后添加我的任意时间量。
下面的代码有效,但在与如此大的数字相乘时可能会溢出。
有没有更好的方法来做到这一点(我得到了以毫秒为单位的时间):
struct timespec ts;
struct timeval now;
gettimeofday(&now, nullptr);
ts.tv_sec = now.tv_sec + milliseconds / 1000;
ts.tv_nsec = now.tv_usec * 1000000000 * (milliseconds % 1000);
ts.tv_sec += ts.tv_nsec / (1000000000);
ts.tv_nsec %= (1000000000);
在上面,我将给定的时间与当前时间相加得到一个绝对时间。
我的替代代码是:
void timeval_to_timespec(struct timeval* tv, struct timespec* ts)
{
ts->tv_sec = tv->tv_sec;
ts->tv_nsec = tv->tv_usec * 1000;
}
struct timespec add_timespec(struct timespec* a, struct timespec* b)
{
struct timespec result = {a->tv_sec + b->tv_sec, b->tv_nsec + b->tv_nsec};
if(result.tv_nsec >= 1000000000)
{
result.tv_nsec -= 1000000000;
++result.tv_sec;
}
return result;
}
//Convert the milliseconds to timespec.
ts.tv_sec = milliseconds / 1000;
ts.tv_nsec = (milliseconds - (ts.tv_sec * 1000)) * 1000000;
//Convert the current time(timeval) to timespec.
timeval_to_timespec(&now, &spec_now);
ts = add_timespec(&ts, &spec_now); //add the milliseconds to the current time.
我想知道是否有更好的方法来完成上述操作。我宁愿不使用我的替代代码,但以前的代码似乎不太安全而且我不喜欢模数。
想法?
你的第一种方法实际上是合理的,除了你在常量方面有一些错别字和错误。
这种方法怎么样:
ts.tv_sec = now.tv_sec + milliseconds / 1000;
ts.tv_nsec = now.tv_usec * 1000 // 1000 ns per us, not a million!
+ (milliseconds % 1000) * 1000000 // a million ns per ms.
ts.tv_sec += ts.tv_nsec / 1000000000;
ts.tv_nsec %= 1000000000;
没有二次加法溢出32位int的危险,因为now.tv_usec * 1000
不超过999,999,000,而(milliseconds % 1000) * 1000000
不超过999,000,000,所以总和最多为1,998,999,000 (最后两行携带的秒数始终为0或1)。
目前我正在尝试获取绝对时间以用于 pthread_mutex_timedlock
。我知道我需要将 timeval
从 gettimeofday
添加到 timespec
,然后添加我的任意时间量。
下面的代码有效,但在与如此大的数字相乘时可能会溢出。
有没有更好的方法来做到这一点(我得到了以毫秒为单位的时间):
struct timespec ts;
struct timeval now;
gettimeofday(&now, nullptr);
ts.tv_sec = now.tv_sec + milliseconds / 1000;
ts.tv_nsec = now.tv_usec * 1000000000 * (milliseconds % 1000);
ts.tv_sec += ts.tv_nsec / (1000000000);
ts.tv_nsec %= (1000000000);
在上面,我将给定的时间与当前时间相加得到一个绝对时间。
我的替代代码是:
void timeval_to_timespec(struct timeval* tv, struct timespec* ts)
{
ts->tv_sec = tv->tv_sec;
ts->tv_nsec = tv->tv_usec * 1000;
}
struct timespec add_timespec(struct timespec* a, struct timespec* b)
{
struct timespec result = {a->tv_sec + b->tv_sec, b->tv_nsec + b->tv_nsec};
if(result.tv_nsec >= 1000000000)
{
result.tv_nsec -= 1000000000;
++result.tv_sec;
}
return result;
}
//Convert the milliseconds to timespec.
ts.tv_sec = milliseconds / 1000;
ts.tv_nsec = (milliseconds - (ts.tv_sec * 1000)) * 1000000;
//Convert the current time(timeval) to timespec.
timeval_to_timespec(&now, &spec_now);
ts = add_timespec(&ts, &spec_now); //add the milliseconds to the current time.
我想知道是否有更好的方法来完成上述操作。我宁愿不使用我的替代代码,但以前的代码似乎不太安全而且我不喜欢模数。
想法?
你的第一种方法实际上是合理的,除了你在常量方面有一些错别字和错误。
这种方法怎么样:
ts.tv_sec = now.tv_sec + milliseconds / 1000;
ts.tv_nsec = now.tv_usec * 1000 // 1000 ns per us, not a million!
+ (milliseconds % 1000) * 1000000 // a million ns per ms.
ts.tv_sec += ts.tv_nsec / 1000000000;
ts.tv_nsec %= 1000000000;
没有二次加法溢出32位int的危险,因为now.tv_usec * 1000
不超过999,999,000,而(milliseconds % 1000) * 1000000
不超过999,000,000,所以总和最多为1,998,999,000 (最后两行携带的秒数始终为0或1)。