本文主要学习三个知识点,第一是UTC时间、GMT时间的概念;第二是在Unix环境下UTC时间与时间戳的转换;第三是在C语言中如何修改时区。
本文参考了《UNP》以及
http://blog.csdn.net/foxir/article/details/43916601
http://blog.csdn.net/ljafl9988/article/details/16847935

一、时间与日期

GMT和UTC

GMT,即格林尼治标准时间,也就是世界时。GMT的正午是指当太阳横穿格林尼治子午线(本初子午线)时的时间。但由于地球自转不均匀不规则,导致GMT不精确,现在已经不再作为世界标准时间使用。

UTC,即协调世界时。UTC是以原子时秒长为基础,在时刻上尽量接近于GMT的一种时间计量系统。为确保UTC与GMT相差不会超过0.9秒,在有需要的情况下会在UTC内加上正或负闰秒。UTC现在作为世界标准时间使用。

所以,UTC与GMT基本上等同,误差不超过0.9秒。

地球自西向东旋转,东边比西边先看到太阳,东边的时间也比西边的早。为了统一世界的时间,1884年的国际经度会议规规定将全球划分为24个时区(东、西各12个时区)。规定英国(格林尼治天文台旧址)为零时区(GMT+00),东1-12区,西1-12区,中国北京处于东8区(GMT+08)。

若中国当前时间为8点整,则英国时间为0点整。

UNIX时间戳

由Unix内核提供的基本时间服务是自 国际标准时间 公元1970年1月1日00:00:00以来的秒数。

二、时间字符串相互转换

时间戳转换为格式化日期

流程: time_t(日历时间) ——->struct tm(以年月日、时分秒表示的时间)——>格式化字符串(利用strftime函数)

#include <stdio.h>  
#include <time.h>  
int main(int argc, const char * argv[])  
    time_t t;  
    struct tm *p;  
    t=0;  
    p=gmtime(&t);  
    char s[100];  
    strftime(s, sizeof(s), "%Y-%m-%d %H:%M:%S", p);  
    printf("%d: %s\n", (int)t, s);  
    return 0;  

结果:
1970-01-01 00:00:00

格式化日期转换为时间戳

流程: struct tm(以年月日、时分秒表示的时间) —> time_t(日历时间)

#include <stdio.h>  
#include <time.h>  
#include <stdlib.h>    
int main(int argc, const char * argv[])  
    struct tm* tmp_time = (struct tm*)malloc(sizeof(struct tm));  
    strptime("19700101080000","%Y%m%d%H%M%S",tmp_time); //时间24时制
    time_t t = mktime(tmp_time);  
    printf("%ld\n",t);  
    free(tmp_time);  
    return 0;  

结果:
0

这里并没有把时间设置为1970年1月1日0点,因为这里涉及到时区的问题。我们目前处于东八区,计算时需要减去8小时。

三、环境变量TZ及时区设置函数

在UNP书里讲过,时间函数除了gmttime()、asctime()不受环境变量TZ的影响外,大部分函数都受到环境变量TZ的影响,这几个函数是: localtime、mktime、ctime和strftime。如果定义了TZ,则这些函数将使用其值以代替系统默认时区。

在Unix环境下可以通过改变系统文件修改环境变量,也可以通过函数setenv()修改。

TZ指定了当前的系统时区。这个时区会影响我们所做的时间转换。例如假设当前的系统时间是8:00AM,如果我们把当前的时区设置成东八区,则标准时间就是(即GMT+0)的时间就是8-8=0:00AM,如果是看成是东6区的话,则标准时间就变成了8-6=2:00AM。

时区设置函数是: setenv(“TZ”, “GMT+0”, 1);//将当前时区设置成标准区
setenv(“TZ”, “GMT-8”, 1); 表示的意思就是:将当前时区设置成东八区。需要注意的是,这里要写成GMT-8才是东八区,不是GMT+8,原因GMT表示时间的方式有点不同,它的格式是(GMT[+/-]offset), 如果目标时区是在东区,则是负的,否则是正的。意思就是本地时区加减多少(offset)才能得到标准时间。

#include <stdio.h>  
#include <time.h>  
#include <stdlib.h>    
int main(int argc, const char * argv[])  
    setenv("TZ", "GMT-8", 1);//时区设置为东8区
    //setenv("TZ", "GMT+0", 1);//将当前时区设置成标准区
    struct tm* tmp_time = (struct tm*)malloc(sizeof(struct tm));  
    strptime("19700101080000","%Y%m%d%H%M%S",tmp_time); //时间24时制
    time_t t = mktime(tmp_time);  
    printf("%ld\n",t);  
    free(tmp_time);  
    return 0;  

当时区设置为东8区时,我们标准时间为8-8=0小时,转换为秒数为:0
当时区设置为国际标准时,标准时间即为8-0=8小时,转换为秒数为:(8-0)*3600=28800

/// 时间相关 /// Add by 成长的小猪(Jason.Song) on 2018/05/10 /// http://blog.csdn.net/jasonsong2008 /// &lt 获取当前的时间的秒数和微秒数本方法需要用到gettimeofday()函数,该函数需要引入的头文件是 sys/time.h 。函数说明int gettimeofday (struct timeval * tv, struct timezone * tz)1、返回值:该函数成功时返回0,失败时返回-1 2、参数 struct timeval{ long tv_sec; //秒 long tv_use... Linux下一般用date 记录当前时间,尤其是我们需要保存测试log的时候,或者设计一个跑多长时间的脚本都需要时间戳。下面看一下平时最常用的几种写法。1 date “+%Y-%m-%d %H:%M” 显示具体时间。5 dmesg log 化为具体时间。date +%s :当前时间时间戳。设计一个iostat监控20秒的脚本。4 利用时间戳 判断脚本的终止时间。2 修改时间 date -s。显示具体时间时间戳。 int tm_sec; /* 秒 – 取值区间为[0,59] */ int tm_min; /* 分 - 取值区间为[0,59] */ int tm_hour; /* 时 - 取值区间为[0,23] */ int tm_mday; /* 一个月中的日期 - 取... 需要引用的头文件请用man文档查询 //arnold add 20220427 修改target时间戳为正确时间戳 start //memset(frm_get->timeStamp, 0, sizeof(unsigned long long)); unsigned long long test_timestamp_ms = 0; struct tm stm = {0}; stm.tm_year=frm_get.dateTime.y Linux C时间函数 time_t struct tm#include关于时间的类型:time_t long型,表示从1970年1月1日到现在经过的秒数。struct tm {int tm_sec; /* 秒 – 取值区间为[0,59] */int tm_min; /* 分 - 取值区间为[0,59] */int tm_hour; /* 时 - 取值区间为[0,... 将Uinx、MySQL时间戳转换为DateTimeusing System;using System.Collections.Generic;using System.Linq;using System.Text;namespace ShopEX_Console.CodesClass{/// /// 将Uninx时间戳转换为DateTime格式/// class UinxTime{/// /// ... 一、问题背景最近项目中需要上传包含时间戳的设备数据到服务器平台。原本想把“年”,“月”,“日”,“时”,“分”, “秒”分别用一个uint8_t的数据类型去存储,即占用6个字节。但是在平台配置协议时,只有一种叫“Unix时间戳”的数据类型。Unix时间戳只占用4个字节,而且Unix时间戳在服务器端更加通用,但是在单片机上没有想Linux环境下现成的time(),localtime(),mktime...