• UNIX系统的显示时间何时会到尽头


    本文转载自:http://www.cnblogs.com/dfcao/p/expertCprogramming_intr0.html

    本文分为三个小块:

    一、UNIX系统中时间的存储形式;

    二、 time_t 的最大值是多少;

    三、 将time_t 的最大值转化为真实世界的时间;

    #---------------------#

    #  欢迎诸位园友指正 #

    #---------------------#

    一、 时间的存储形式

    UNIX下存储时间常见的有两种存储方式:

    一种是time_t 这种类型,存储了从1970年到现在经过了多少秒,在UNIX系统中,time_t 是 long 类型的typedef 形式,它的定义位于文件/usr/include/time.h中。要想更精确一点,可以用结构体 struct timeval,它精确到微秒,见下列代码。

    struct timeval
    {
        long tv_sec;   /*秒*/
        long tv_usec;  /*微秒*/
    };

    另一种是用一个结构体 struct tm 来分别存储年月日时分秒的,见下列代码。

    复制代码
    struct tm
    {
        int tm_sec;  /*秒,正常范围0-59, 但允许至61*/
        int tm_min;  /*分钟,0-59*/
        int tm_hour; /*小时, 0-23*/
        int tm_mday; /*日,即一个月中的第几天,1-31*/
        int tm_mon;  /*月, 从一月算起,0-11*/int tm_year; /*年, 从1900至今已经多少年*/int tm_wday; /*星期,一周中的第几天, 从星期日算起,0-6*/
        int tm_yday; /*从今年1月1日到目前的天数,范围0-365*/
        int tm_isdst;/*日光节约时间的旗标*/
    };
    复制代码

    需要特别注意的是,年份是从1900年起至今多少年,而不是直接存储如2011年,月份从0开始的,0表示一月,星期也是从0开始的, 0表示星期日,1表示星期一。

    二、 time_t 的最大值是多少?

    这个问题不难。由于time_t 是 long 类型的typedef 形式,所以time_t的最大值也即 long 类型的最大值;

    对于32位操作系统,long类型的最大值为0x7FFFFFFF,转化为十进制约为21亿,而unsigned long的最大值约为42亿。见下列代码。

    复制代码
    #include<iostream>
    #include<time.h>
    #include<iomanip>
    using namespace std;
    
    int main(){
      long long_max = 0x7FFFFFFF;
      cout << dec << long_max << endl;   //以十进制输出long_max的值,输出: 2147483647
      
      time_t time_t_max = 0x7FFFFFFF;
      cout << dec << time_t_max << endl; //输出: 2147483647
      
      unsigned long unsigned_long_max = 0xFFFFFFFF;
      cout << dec << unsigned_long_max << endl; //输出: 4294967295
    }
    复制代码

    三、 time_t 的最大值转化为时间

    time_t的最大值已经知道了,那么如何将此最大值转化为时间呢? 库time.h 已经为我们准备了很多函数来实现这个目的。

    常用的时间函数:
    #include <time.h>

    char *ctime(const time_t *timep);

    将time_t 类型转换为真实世界的时间,以字符串显示,得到的时间是经过时区转换的时间。

     

    char *asctime(const struct tm *timeptr);

    将struct tm 类型转换为真实世界的时间,以字符串的形式显示,它和ctime不同在于: 1,传入的参数形式不一样; 2,不经过时区转换。

    struct tm* gmtime(const time_t *timep);

    将time_t 类型转换为一个struct tm 类型,得到的是没有经过时区转换的UTC时间(译为世界标准时间 or 世界协调时间)。

     

    stuct tm* localtime(const time_t *timep);

    将time_t 类型转换为一个struct tm 类型,和gmtime类似,但是它是经过时区转换的时间,得到的是本地时间。

    有了这些函数,我们就可以将time_t 类型的最大值转换为真实世界的时间了。

    最简单的一个方案: 用ctime() 函数将time_t 转换为真实时间,见下代码。

    复制代码
    #include<stdio.h>
    #include<time.h>int main() {
      time_t time_t_max = 0x7FFFFFFF;  
      
      printf("time_t_max = %s 
    ", ctime(&time_t_max)); 
      //输出: time_t_max = Tue Jan 19 11:14:07 2038
    
      return 0;  
    }
    复制代码

    也就是说32位UNIX计算机能够显示的时间最多到2038年1月19号11点14分07秒(北京时间)。但是这里有一个问题,ctime() 函数把参数转换为当地时间,它跟世界标准时间UTC并不一致,而我们要得到的是世界通用的UTC时间。

    修订后方案: 用gmtime() 函数将time_h 的最大值转换成UTC时间值,再将这个strct tm 类型的数据用asctime() 函数转换成真是世界的时间。见下代码。

    复制代码
    #include<stdio.h>
    #include<time.h>
    
    int main() {
      time_t time_t_max = 0x7FFFFFFF;  
      printf("time_t_max = %s 
    ", asctime(gmtime(&time_t_max))); 
      //输出: time_t_max = Tue Jan 19 03:14:07 2038
      return 0;  
    }
    复制代码

    得到结果2038年1月19号03点14分07秒(UTC时间),这与北京时间有八小时的时差。

    也就是说,到2038年1月19号03点14分07秒(UTC时间)这个点,32位计算机在UNIX平台存放的时间变量将溢出,超过此一瞬间,时间将会“绕回”且在内部被表示为一个负数,并造成程序无法工作,因为它们无法将此时间识别为2038年,而可能会依个别实现而跳回1970年或1901年,错误的计算及动作可能因此产生。这就是著名的2038年问题。直到2006年,仍然有数以亿计的32位系统在运行中,特别是许多嵌入式系统。相对于一般计算机科技18至24个月的革命性更新,嵌入式系统可能直至使用寿命终结都不会改变。

    当然了,64位操作系统可以记录至约2900亿年后的292,277,026,596年12月4日15:30:08,星期日(UTC),完全不存在这个2038年问题。

    #---------------------------------------------------------------------------------#

    参考文献

    《C专家编程》, Peter Van Der Linden 著, 徐波 译  

    cousera course: Introduction to Computing, by Li Ge

    "c++ 时间类型详解 time_t", by kakaka2011

    "2038年问题", by 维基百科

  • 相关阅读:
    Oracle根据两点经纬度计算距离(转载)
    TCP小见解
    git describe功能实现
    UE中基本图形的原始大小是多大
    SQL多行合并与HTML组装,不转义特殊字符
    磁盘空间不足引起ftp报"553 Could not create file"
    一键安装包安装lnmp
    宝塔面板(Linux版)安装与使用
    Redhat7-yum本地源安装配置
    Oracle数据库多个表空间使用情况查询
  • 原文地址:https://www.cnblogs.com/zzb-Dream-90Time/p/5840051.html
Copyright © 2020-2023  润新知