美文网首页
7个设置/获取接口了解Linux时间管理

7个设置/获取接口了解Linux时间管理

作者: 开源519 | 来源:发表于2024-11-23 21:36 被阅读0次

7个设置/获取接口了解Linux时间管理

[TOC]

引言

  最近的项目开发中,频繁遇到了时间戳相关的问题,如时间回退至1970年、时区错误及时间同步不准确等。鉴于此前仅对时间接口的使用有所了解而未深入探究其原理,本篇文章进行一次系统性整理,以便后续参考。文章若存在一些错误,可在留言区明确指出。

<span style="font-size: 12px;">
<span style="color: blue;"> 注:文末提供本文源码获取方式。文章不定时更新,喜欢本公众号系列文章,可以星标公众号,避免遗漏干货文章。源码开源,如果对您有帮助,帮忙分享、点赞加收藏喔!</span>
</span>

基础概念

Linux 中的时间形式主要以两种形式呈现:

  • 相对时间 指相对于某个基准点来衡量时间流逝。通常用于描述进程运行的时间或两个事件之间的时间差。
    • 进程时间
      即进程消耗的时间,包含用户空间代码运行的时间和在内核在该进程消耗的时间(不包括进程被挂起或停止的时间)。
    • 单调时间
      是一种始终递增的时间计数器,不受系统时钟调整的影响,常用于计算程序内部的持续时间。
  • 绝对时间 指具体的日期和时刻,它与地球上的特定时间标准相关联。
    • GMT(Greenwich Mean Time 格林威治时间)
      基于英国伦敦附近的格林尼治天文台的本初子午线的标准时间
    • UTC(Universal Time Coordinated 世界标准时间)
      一种国际标准时间,与GMT几乎相同,但更精确,用于避免地球自转速度变化带来的影响
    • 本地时间
      根据用户所在地理位置所采用的时间,会随地理位置的不同而有所差异,同时也会受到夏令时等因素的影响

相关结构体

  时间编程中常用要用到的时间结构体有time_ttimevaltimespectm。《Unix环境高级编程》中一张图准确的反应出time_ttm之间的关系:

时间函数之间的关系.png
  • time_t:最简单的数据湖结构,表示从1970年1月1日00:00:00 UTC到现在的秒数。
  • tm:包含日期和时间的具体组成部分(年、月、日、时、分、秒等),通常由 time_t 转换而来,用于显示或解析时间。
  • timeval:微秒级精度,包含秒(tv_sec)和微秒(tv_usec)。
  • timespec:纳秒级精度,包含秒(tv_sec)和纳秒(tv_nsec)。
  • clock_t:表示程序执行过程中消耗的CPU时间,单位是 CLOCKS_PER_SEC

相关函数

时间获取

  • time:
    • 函数原型:time_t time(time_t *tloc);
    • 功能描述:该函数返回从1970年1月1日00:00:00 UTC以来的秒数。如果tloc不是NULL,则返回的时间值也会存储在tloc指向的位置。
    • 返回值:成功时返回当前时间(以秒为单位),失败时返回(time_t)(-1)
  • gettimeofday:
    • 函数原型:int gettimeofday(struct timeval *tv, struct timezone *tz);
    • 功能描述:这个函数提供了比time()更高的精度,可以获取当前时间精确到微秒。struct timeval包含两个成员:tv_sec(秒数)和tv_usec(微秒数)。struct timezone已经废弃,通常传入NULL
    • 返回值:成功时返回0,出错时返回-1,并设置errno
  • clock_gettime:
    • 函数原型:int clock_gettime(clockid_t clk_id, struct timespec *tp);
    • 功能描述:此函数提供了更高的时间分辨率,可以获取纳秒级别的精度。struct timespec包含两个成员:tv_sec(秒数)和tv_nsec(纳秒数)。clk_id参数指定了要查询的时间源(带有“可选”指并非所有系统都必须支持):
      • CLOCK_REALTIME
        描述:系统实时钟,反映当前的实际时间。
        特点:受系统时间调整的影响。
      • CLOCK_MONOTONIC
        描述:单调时钟,从某个未指定的起点开始计时。
        特点:不受系统时间调整的影响,适合用于测量时间间隔。
      • CLOCK_PROCESS_CPUTIME_ID
        描述:当前进程的CPU时间。
        特点:包括用户态和内核态的CPU时间。
      • CLOCK_THREAD_CPUTIME_ID
        描述:当前线程的CPU时间。
        特点:仅包括当前线程的CPU时间。
      • CLOCK_MONOTONIC_RAW (可选)
        描述:高精度单调时钟,不受系统时间调整的影响。
        特点:提供更高的时间分辨率。
      • CLOCK_REALTIME_COARSE (可选)
        描述:较低精度的系统实时钟。
        特点:速度快,但精度较低。
      • CLOCK_MONOTONIC_COARSE (可选)
        描述:较低精度的单调时钟。
        特点:速度快,但精度较低。
    • 返回值:成功时返回0,出错时返回-1,并设置errno
  • times:
    • 函数原型:clock_t times(struct tms *buf);
    • 功能描述:此函数用于获取进程所使用的时间信息,包括用户态和内核态下的运行时间。struct tms包含四个成员:tms_utime(用户态运行时间)、tms_stime(内核态运行时间)、tms_cutime(子进程用户态运行时间)、tms_cstime(子进程内核态运行时间),所有时间都以时钟滴答数(clock ticks)表示。
    • 返回值:成功时返回进程自开始执行以来所使用的时钟滴答数,若出错则返回-1L

时间设置

  • stime:
    • 函数原型:int stime(const time_t *t);
    • 功能描述:此函数用于将系统的实时钟设置为指定的时间。t是一个指向time_t类型变量的指针,该变量包含了自1970年1月1日00:00:00 UTC以来的秒数。
    • 返回值:成功时返回0,失败时返回-1,并设置errno
    • 注意事项:stime()函数通常需要root权限才能执行,且至 Linux 2.6.x之后版本不推荐使用,本地glibc 2.35实测已无法编译此函数。
  • settimeofday:
    • 函数原型:int settimeofday(const struct timeval *tv, const struct timezone *tz);
    • 功能描述:此函数允许设置系统的实时时间和时区信息。tv指向一个struct timeval结构,该结构包含了秒数和微秒数,用来表示新的系统时间。tz指向一个struct timezone结构,该结构包含了分钟偏移量和夏令时标志位,不过在现代系统中,通常不需要设置时区信息,因此可以传递NULL
    • 返回值:成功时返回0,失败时返回-1,并设置errno
    • 注意事项:与stime()类似,settimeofday()也需要适当的权限才能改变系统时间。
  • clock_settime:
    • 函数原型:int clock_settime(clockid_t clk_id, const struct timespec *tp);
    • 功能描述:此函数用于设置由clk_id标识的时钟。tp指向一个struct timespec结构,该结构包含了秒数和纳秒数,可以用来非常精确地设置时间。通常只允许设置时间源CLOCK_REALTIME(系统实时钟)。
    • 返回值:成功时返回0,失败时返回-1,并设置errno
    • 注意事项:修改系统实时钟通常需要root权限,而其他类型的时钟通常不允许设置。

时间转换

  • asctime / asctime_r(tm -> char*)
    • 函数原型:char *asctime(const struct tm *timeptr); / char *asctime_r(const struct tm *timeptr, char *buf);
    • 功能描述:将 struct tm 结构转换为字符串格式,格式为 "Sun Sep 16 01:03:52 1979\n"。asctime_r 是线程安全版本。
    • 返回值:返回指向字符串的指针。
    • 注意事项:asctime 返回的字符串是静态分配的,多次调用会覆盖前一次的结果。
  • mktime (tm -> time_t)
    • 函数原型:time_t mktime(struct tm *timeptr);
    • 功能描述:将 struct tm 结构转换为 time_t 类型的时间值。
    • 返回值:成功时返回 time_t 类型的时间值,失败时返回 (time_t)(-1)
    • 注意事项:mktime 可能会修改传入的 struct tm 结构中的某些字段。
  • ctime / ctime_r (time_t -> char*)
    • 函数原型:char *ctime(const time_t *timep); / char *ctime_r(const time_t *timep, char *buf);
    • 功能描述:将 time_t 类型的时间值转换为字符串格式,格式为 "Sun Sep 16 01:03:52 1979\n"。ctime_r 是线程安全版本。
    • 返回值:返回指向字符串的指针。
    • 注意事项:ctime 返回的字符串是静态分配的,多次调用会覆盖前一次的结果。
  • gmtime / gmtime_r (time_t -> tm ) UTC
    • 函数原型:struct tm *gmtime(const time_t *timep); / struct tm *gmtime_r(const time_t *timep, struct tm *result);
    • 功能描述:将 time_t 类型的时间值转换为 UTC 时间的 struct tm 结构。gmtime_r 是线程安全版本。
    • 返回值:成功时返回指向 struct tm 结构的指针,失败时返回 NULL
    • 注意事项:gmtime 返回的 struct tm 结构是静态分配的,多次调用会覆盖前一次的结果。
  • localtime / localtime_r (time_t -> tm) 本地时间
    • 函数原型:struct tm *localtime(const time_t *timep); / struct tm *localtime_r(const time_t *timep, struct tm *result);
    • 功能描述:将 time_t 类型的时间值转换为本地时间的 struct tm 结构。localtime_r 是线程安全版本。
    • 返回值:成功时返回指向 struct tm 结构的指针,失败时返回 NULL
    • 注意事项:localtime 返回的 struct tm 结构是静态分配的,多次调用会覆盖前一次的结果。
  • difftime (time_t -> double)
    • 函数原型:double difftime(time_t time1, time_t time0);
    • 功能描述:计算两个 time_t 类型的时间值之间的差值,以秒为单位。
    • 返回值:返回两个时间值之间的差值,以秒为单位。

时间格式化

  • strftime (tm -> char*)
    • 函数原型:size_t strftime(char *str, size_t maxsize, const char *format, const struct tm *timeptr);
    • 功能描述:根据指定的格式字符串 formatstruct tm 结构转换为字符串,并存储在 str 中。最多写入 maxsize 个字符(包括终止符 \0)。
    • 返回值:成功时返回实际写入的字符数(不包括终止符 \0),如果缓冲区太小无法容纳结果,则返回 0
    • 注意事项:确保提供的缓冲区 str 足够大,以避免溢出。

时区设置

  时区会影响到本地时间与UTC时间之间的转换(即本地时间 = UTC + 时区)。
  查阅了一些文档,目前Ubuntu上时区记录在路径/etc/localtime,其通常为软链接,指向具体的时区文件,例如 /etc/localtime -> /usr/share/zoneinfo/Asia/Shanghai。通过修改/etc/localtime指向即可修改为对应的时区(/etc/timezone也会记录当前时区,但似乎仅用于显示)。

实例测试

测试time/stime

time

void TestGetTime()
{
    // time UTC时间戳
    time_t tmt1 = time(NULL);
    printf("timestamp  : %ld\n", tmt1);

    // ctime_r UTC时间戳转换为本地时间字符串
    char cbuf[50] = {0};
    ctime_r(&tmt1, cbuf);
    printf("ctime_r    : %ld(%6d) %s", tmt1, 0, cbuf);

    // gmtime_r UTC时间戳转换为UTC时间字符串
    tm gtm;
    time_t tmt2;
    char gbuf[50] = {0};
    gmtime_r(&tmt1, &gtm);
    asctime_r(&gtm, gbuf);
    tmt2 = mktime(&gtm);    // mktime 会自动减时区
    printf("gmtime_r   : %ld(%6ld) %s %s", tmt2, tmt2-tmt1, gtm.tm_zone, gbuf);

    // 将时间戳转换为本地时间
    tm ltm;
    time_t tmt3;
    char lbuf[50] = {0};
    localtime_r(&tmt1, &ltm);
    asctime_r(&ltm, lbuf);
    tmt3 = mktime(&ltm);
    printf("localtime_r: %ld(%6ld) %s %s", tmt3, tmt3-tmt1, ltm.tm_zone, lbuf);

    char buf3[50] = {0};
    strftime(buf3, 50, "%Z %a %b %d %H:%M:%S %Y", &ltm);
    printf("strftime   : %ld(%6ld) %s\n", tmt3, tmt3-tmt1, buf3);
}

测试结果

timestamp  : 1732450363
ctime_r    : 1732450363(     0) Sun Nov 24 20:12:43 2024
gmtime_r   : 1732421563(-28800) CST Sun Nov 24 12:12:43 2024
localtime_r: 1732450363(     0) CST Sun Nov 24 20:12:43 2024
strftime   : 1732450363(     0) CST Sun Nov 24 20:12:43 2024

gmtime_r 打印的是UTC时间戳,与本地时间相差28800s (8h),即本地与UTC时间相差8h

测试gettimeofday/settimeofday

void Testgettimeofday()
{
    struct timeval tv;
    gettimeofday(&tv, NULL);
    printf("tv_sec: %ld, tv_usec: %ld\n", (long)tv.tv_sec, (long)tv.tv_usec);
}

void Testsettimeofday()
{
    Testgettimeofday();
    struct timeval tv1;
    tv1.tv_sec = 1731985300;
    tv1.tv_usec = 100;
    int ret = settimeofday(&tv1, NULL);
    if (ret == -1) {
        perror("settimeofday");
    }
    Testgettimeofday();
}

测试结果

tv_sec: 1732450828, tv_usec: 890873
tv_sec: 1731985300, tv_usec: 150

注意在调用设置时间接口时,需要root权限执行,否则会设置失败。

测试clock_gettime/clock_settime

void Testclock_gettime()
{
    std::string name[] = {
        "CLOCK_REALTIME",
        "CLOCK_MONOTONIC",
        "CLOCK_PROCESS_CPUTIME_ID",
        "CLOCK_THREAD_CPUTIME_ID",
        "CLOCK_MONOTONIC_RAW",
        "CLOCK_REALTIME_COARSE",
        "CLOCK_MONOTONIC_COARSE",
        "CLOCK_BOOTTIME",
        "CLOCK_REALTIME_ALARM",
        "CLOCK_BOOTTIME_ALARM",
    };

    // printf("Test clock_gettime\n");
    printf("%-25s  %10s  %10s\n", "CLOCK TYPE", "SEC", "NSEC");
    printf("-----------------------------------------------------------------------------\n");
    for (int i = 0; i <= CLOCK_BOOTTIME_ALARM; i++) {
        struct timespec ts;
        clock_gettime(i, &ts);
        printf("%-25s: %10ld, %10ld\n", name[i].c_str(), (long)ts.tv_sec, (long)ts.tv_nsec);
    }
    printf("-----------------------------------------------------------------------------\n");
}

void Testclock_settime()
{
    Testclock_gettime();

    // Only CLOCK_REALTIME is allowed to be set
    struct timespec ts1;
    ts1.tv_sec = 1731985300;
    ts1.tv_nsec = 100;
    int ret = clock_settime(CLOCK_REALTIME, &ts1);
    if (ret == -1) {
        perror("clock_settime");
    }

    Testclock_gettime();
}

测试结果

CLOCK TYPE                        SEC        NSEC
-----------------------------------------------------------------------------
CLOCK_REALTIME           : 1732451153,  160842537
CLOCK_MONOTONIC          :      45250,  516265743
CLOCK_PROCESS_CPUTIME_ID :          0,     908800
CLOCK_THREAD_CPUTIME_ID  :          0,     910400
CLOCK_MONOTONIC_RAW      :      45249,   35729391
CLOCK_REALTIME_COARSE    : 1732451153,  145187465
CLOCK_MONOTONIC_COARSE   :      45250,  500594052
CLOCK_BOOTTIME           :      45250,  516287258
CLOCK_REALTIME_ALARM     : 1732451153,  160881972
CLOCK_BOOTTIME_ALARM     :      45250,  516289642
-----------------------------------------------------------------------------
CLOCK TYPE                        SEC        NSEC
-----------------------------------------------------------------------------
CLOCK_REALTIME           : 1731985300,      32053
CLOCK_MONOTONIC          :      45250,  516347493
CLOCK_PROCESS_CPUTIME_ID :          0,     988400
CLOCK_THREAD_CPUTIME_ID  :          0,     989400
CLOCK_MONOTONIC_RAW      :      45249,   35795309
CLOCK_REALTIME_COARSE    : 1731985300,        100
CLOCK_MONOTONIC_COARSE   :      45250,  516314680
CLOCK_BOOTTIME           :      45250,  516352354
CLOCK_REALTIME_ALARM     : 1731985300,      38528
CLOCK_BOOTTIME_ALARM     :      45250,  516353885
-----------------------------------------------------------------------------

从测试结果看,更改系统时间时,仅有时间源CLOCK_REALTIMECLOCK_REALTIME_ALARM会随之修改而跳变,其他时间源不会随着系统时间的修改而跳变。在了解这些特性后,在编写应用程序时选择合适的时间源,以满足不同的需求。

测试sleep后,时间的变化

void TestTimeWithSleep(int sec)
{
    std::string name[] = {
        "CLOCK_REALTIME",
        "CLOCK_MONOTONIC",
        "CLOCK_PROCESS_CPUTIME_ID",
        "CLOCK_THREAD_CPUTIME_ID",
        "CLOCK_MONOTONIC_RAW",
        "CLOCK_REALTIME_COARSE",
        "CLOCK_MONOTONIC_COARSE",
        "CLOCK_BOOTTIME",
        "CLOCK_REALTIME_ALARM",
        "CLOCK_BOOTTIME_ALARM",
    };

    struct timespec ots[10];
    for (int i = 0; i < 10; i++) {
        clock_gettime(i, &ots[i]);
    }

    sleep(sec);
    struct  timespec nts[10];
    for (int j = 0; j < 10; j++) {
        clock_gettime(j, &nts[j]);
    }

    printf("%-25s  %10s %10s %10s %10s %7s %8s\n", "CLOCK TYPE", "OLDSEC", "OLDNSEC", "NEWSEC", "NEWNSEC", "DIFFSEC", "DIFFNSEC");
    printf("-------------------------------------------------------------------------------------------\n");
    for (int i = 0; i <= CLOCK_BOOTTIME_ALARM; i++) {
        printf("%-25s: %10ld %10ld %10ld %10ld %7ld %8ld\n",
        name[i].c_str(), (long)ots[i].tv_sec, (long)ots[i].tv_nsec,
        (long)nts[i].tv_sec, (long)nts[i].tv_nsec, (long)(nts[i].tv_sec - ots[i].tv_sec), (long)(nts[i].tv_nsec - ots[i].tv_nsec));
    }
}

测试结果

sleep 5s 结果如下:

CLOCK TYPE                     OLDSEC    OLDNSEC     NEWSEC    NEWNSEC DIFFSEC DIFFNSEC
-------------------------------------------------------------------------------------------
CLOCK_REALTIME           : 1732451618  944834581 1732451623  945683524       5   848943
CLOCK_MONOTONIC          :      45716  300258307      45721  301107230       5   848923
CLOCK_PROCESS_CPUTIME_ID :          0    1010700          0    1048800       0    38100
CLOCK_THREAD_CPUTIME_ID  :          0    1011000          0    1050000       0    39000
CLOCK_MONOTONIC_RAW      :      45714  819871428      45719  820723025       5   851597
CLOCK_REALTIME_COARSE    : 1732451618  935986823 1732451623  935984705       5    -2118
CLOCK_MONOTONIC_COARSE   :      45716  291410495      45721  291408377       5    -2118
CLOCK_BOOTTIME           :      45716  300260726      45721  301110251       5   849525
CLOCK_REALTIME_ALARM     : 1732451618  944837451 1732451623  945687049       5   849598
CLOCK_BOOTTIME_ALARM     :      45716  300287520      45721  301111127       5   823607

  从上述结果看,CLOCK_PROCESS_CPUTIME_IDCLOCK_THREAD_CPUTIME_ID没有记录sleep 5s的时间,也应征了上述所描述的进程挂起或停止时,进程时间不会记录。
  用times接口验证会更明显,sleep前后times获取的时间值基本没有变化。

测试修改时区

void TestSetTimeZone(const std::string& tz)
{
    int ret = 0;
    std::string target = "/usr/share/zoneinfo/" + tz;

    ret = unlink("/etc/localtime");
    if (ret == -1) {
        perror("unlink");
    }

    ret = symlink(target.c_str(), "/etc/localtime");
    if (ret == -1) {
        perror("symlink");
        return;
    }

    tzset();
    TestGetTimeZone();
    TestGetTime();
}

测试结果

设置时区America/New_York

timestamp  : 1732452775
ctime_r    : 1732452775(     0) Sun Nov 24 07:52:55 2024
gmtime_r   : 1732470775( 18000) EST Sun Nov 24 12:52:55 2024
localtime_r: 1732452775(     0) EST Sun Nov 24 07:52:55 2024
strftime   : 1732452775(     0) EST Sun Nov 24 07:52:55 2024

通过打印可看出时区已经显示EST,与Asia/Shanghai时区相差了13h。

总结

  • Linux 时间相关接口比较简单,之前没有系统了解过,一直使用的比较混乱,其实主要就是根据实际的精度需求选择对应的接口即可。
  • 在嵌入式开发项目中,时区管理是一项不可忽视的任务。通常情况下,通过GPS基站或网络时间协议(NTP)服务器进行时间同步以确保设备时区的准确性。在调整时区时,推荐仅更新系统的时区配置文件,而不是直接对系统时间进行增减操作,以此避免可能的时间计算错误。
  • 在实际项目中,推荐使用协调世界时(UTC)作为时间基准,而非依赖于本地时间。这是因为本地时间会因时区变更而发生变化,而UTC提供了一个全球统一的标准,不受地理位置的影响。
  • 时间服务是操作系统中的基础组成部分之一,因此在进行时间校准时,需要仔细规划校准的时间点。不恰当的时间跳跃可能导致依赖于系统时间的应用程序和服务出现故障。
  • 在过去的经验中,wait_for会随着时间跳变而异常。尽管印象中,不应该这样,其依赖的应该是相对时间即单调时间。经过查阅相关资料,发现gcc版本和glibc版本对wait_for都有影响,gcc >=10 且 glibc >= 2.30 才会对程序行为没有影响。

相关文章

  • NODEJS工程-通用GET方法

    。智能获取方法GET每个表都自动有的 (主要作用是将获取数据的许多设置交给前端减少接口修改次数) 。前端应了解只获...

  • k8s资源对象api获取数据

    概述 实现方式 1.获取api 结果 : 2.设置代理 3.获取接口数据

  • 测试数据获取操作

    测试数据的获取操作通过去设置参数和变量进行提交和操作。在做接口测试的时候,我们经常需要去了解接口的入参和出参。每一...

  • 静态ip配置中的问题

    一.需改Linux中的网络接口配置 设置dns 重启网络服务

  • JAVA 8 Future

    Future本身是一个接口,提供取消、查询取消、查询完成、获取结果、设置超时获取结果五个接口方法;默认提供了Pen...

  • vmware linux 配置网络

    第一步:设置vmnet8 的ip为自动获取 如下所示:设置 自动获取网络ip地址 第二步:linux 网络配置 设...

  • shell 常用功能函数

    参考书籍 Linux shell 脚本攻略 目录 使用别名 获取终端信息 获取、设置日期和延时 调试脚本 函数和参...

  • Linux DRM那些事-HDMI接口EDID获取

    本文基于RockPI 4A单板Debian系统Linux 4.4内核DRM框架介绍HDMI接口 EDID获取方法。...

  • 公众号配置

    一:设置 JS接口安全域名 和 网页授权域名 二:设置获取access_token白名单 IP:115.29.22...

  • 接口报错:could not validate certific

    1.我再调用接口的时候出现这个错误,原因是平板没有设置自动获取时间,手动设置的2010。把自动获取时间打开就OK了

网友评论

      本文标题:7个设置/获取接口了解Linux时间管理

      本文链接:https://www.haomeiwen.com/subject/sfoajjtx.html