Linux:获取高精度时间
创始人
2025-05-29 03:26:04
0

        在linux下有很多获取时间的函数,不过大部分都是需要调用内核,对于性能要求非常高的程序可能无法满足要求,需要特殊的方法替代常见的api。

1、time localtime

time函数,获取从1970到现在的秒数,精确度只有秒,通过localtime转换

time_t testtm;
time(&testtm);
testtm = time(NULL);
struct tm *ptm ;
ptm = localtime(&testtm);
struct tm ptm = { 0 }; 
localtime_r(&testtm, &ptm); 

        注意:localtime不是线程安全的,替代函数localtime_r用来在多线程之间使用,是线程安全的。localtime是不可重入的,多次调用会被覆盖。从其返回值也可以看出,localtime返回一个指针,所以这个指针的数据存储就成了大问题,导致多线程访问同一个,多次调用会覆盖。而localtime_r传递一个结构体,会帮你复制,就没有相互影响的问题。所以不要用localtime就对了,避免各种意想不到的问题。

        不可以在信号响应函数中使用localtime或localtime_r,会导致卡死。

2、gettimeofday

        这个是获取毫秒级别的api

3、clock_gettime

        这个可以返回纳秒级别的数据,并且可以返回具体的系统时间(也是从1970年到现在的秒数)或者系统从开机开始到现在的运行时间。

struct timespec mtime;
clock_gettime(CLOCK_MONOTONIC, &mtime);
tm nowTime;
localtime_r(&mtime.tv_sec, &nowtime);

CLOCK_REALTIME和CLOCK_MONOTONIC分别对应是具体时间,还是机器运行时间。

4、rdtsc

        rdtsc是cpu运行周期计数,只在Intel的x86_64架构上才可以使用,其他的cpu架构请查询其他用法。这个数值与cpu的频率有关,1Ghz等于十亿赫兹(1,000,000,000 Hz),表示1秒中计数增加十亿次。

uint64_t get_tsc()
{
    uint64_t mlow, mhigh;
    __asm__ volatile("rdtsc" : "=a"(mlow), "=d"(mhigh));
    return (mhigh << 32) | mlow;
}

        这个方法会通过汇编,把当前cpu计数(64位)高32位存放到一个寄存器,低32位存放到另一个寄存器,我们取出后使用。
        记住,如果自己修改,汇编那一行=a和=d是不可以变的,只能替换我们定义的变量mlow和mhigh名称。

        这个方式不仅性能好,而且精度高,因为其没有经过内核,比上面的clock_gettime性能要好,不过比time localtime差一点,毕竟time localtime的精度太低了。
        这个的用法一般是程序开始,sleep一段时间,1秒或者10秒,然后计算这个值,就直到当前系统具体1秒或者1毫秒等累加多少计数。

        注意:rdtsc是针对cpu运行频率的,但是同一个电脑上的cpu,不同内核,可能cpu频率并不一致,所以这个值最好不要跨cpu内核使用
        其次,同一个cpu内核,不同时间,其频率也会变化,所以需要锁定cpu的频率。

5、constant_tsc

为了解决上面rdtsc的问题,新的内核提供了这个参数,表示你可以随便使用,不用再担心不同核心差异化导致的计算值不一致问题。
具体系统是否支持,可以使用cat /proc/cpuinfo|grep constant_tsc查看一下。

6、rdtscp

使用rdtsc或者constant_tsc还会有一个问题,就是由于指令优化,可能会出现乱序,导致不准确,如果你对时间要求非常高,可以使用rdtscp

7、总结

大概效率
time(5.65) > rdtsc(8.81) > rtdscp(15.95) > gettimeofday(22.70) > clock_gettime(23.87)

相关内容

热门资讯

《JavaEE》进程调度的基本... ps:上一篇的知识没有讲全 此篇为补充~  目录 什么是进程? 进程...
小白学Pytorch系列--T... 小白学Pytorch系列–Torch API (9) Other Operations atlea...
Java 注解(详细学习笔记) 注解 注解英文为Annotation Annotation是JDK5引入的新的技术 Annota...
00、数据结构绪论 00、数据结构绪论 数据是信息的载体 数据元素:每一个个体,往下拆分是不...
【党建文化】粽香传情,法润人心... 端午临仲夏,时清日复长。在端午佳节来临之际,为弘扬中华优秀传统文化,深化党群共建工作,传递律所人文关...
上海金融监管局:持续稳步推进金... 钛媒体App 5月30日消息,上海金融监管局副局长王鑫泽在上海市政府新闻发布会上表示,近年来上海金融...
绿盟科技:公司按照相关法律法规... 证券之星消息,绿盟科技(300369)05月30日在投资者关系平台上答复投资者关心的问题。 投资者提...
航天信息:公司未来如涉及相关事... 证券之星消息,航天信息(600271)05月30日在投资者关系平台上答复投资者关心的问题。 投资者提...
84、Latent-NeRF ... 简介 论文:https://arxiv.org/abs/2211.07600 dre...
栈(数据结构系列7) 前言: 这节中小编带你了解数据结构中的栈结构和队列结构,以及分别自己模拟...