C/C++标准库之转换UTC时间到local本地时间详解

时间:2022-10-17 12:07:53

前言

UTC 时间DateTime.UtcNow 和 系统本地时间 DateTime.Now 相差8个时区 ,美国本地时间和北京时间相差15个时区: 美国,而一般使用UTC时间方便统一各地区时间差异。

场景

      1.如果有面向全球用户的网站, 一般在存储时间数据时存储的是UTC格式的时间, 这样时间是统一的, 并可以根据当地时区来进行准确的转换.

      2.存储本地时间的问题就在于如果换了时区, 那么显示的时间并不正确. 所以我们存储时间时最好还是存储UTC时间,便于正确的转换.

说明

1.C/C++标准库提供了标准函数可以转换, 不需要借助Win32 API.

例子

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
// test_datetime_format.cpp : 定义控制台应用程序的入口点。
//
 
#include "stdafx.h"
#include <time.h>
#include <sstream>
#include <iostream>
#include <assert.h>
 
//2014-09-13T10:52:36Z
//2014-09-13 10:52:36
char* ConvertUtcToLocalTime(struct tm* t2,const char* date){
 
 struct tm t;
 memset(&t,0,sizeof(t));
 t.tm_year = atoi(date)-1900;
 t.tm_mon = atoi(date+5)-1;
 t.tm_mday = atoi(date+8);
 t.tm_hour = atoi(date+11);
 t.tm_min = atoi(date+14);
 t.tm_sec = atoi(date+17);
 
 time_t tt = _mkgmtime64(&t);
 if(tt != -1){
 if(t2 == NULL){
  t2 = &t;
 }
 *t2 = *localtime(&tt);
 char* ds = (char*) malloc(24);
 memset(ds, 0, 24);
 sprintf(ds, "%.4d-%.2d-%.2d %.2d:%.2d:%.2d", t2->tm_year + 1900,
  t2->tm_mon + 1, t2->tm_mday, t2->tm_hour, t2->tm_min,
  t2->tm_sec);
 return ds;
 }
 return NULL;
}
 
 
//https://www.w3.org/TR/NOTE-datetime
//https://msdn.microsoft.com/en-us/library/2093ets1.aspx
//2014-09-13T10:52:36Z
int _tmain(int argc, _TCHAR* argv[])
{
 const char* kTime = "2014-09-13 18:52:36";
 std::cout << "Source DateTime: " << "2014-09-13T10:52:36Z" << std::endl;
 auto t = ConvertUtcToLocalTime(NULL,"2014-09-13T10:52:36Z");
 std::cout << "Dest DateTime: " << t << std::endl;
 assert(!strcmp(t,kTime));
 
 t = ConvertUtcToLocalTime(NULL,"2014-09-13 10:52:36");
 std::cout << t << std::endl;
 assert(!strcmp(t,kTime));
 
 struct tm tt;
 t = ConvertUtcToLocalTime(&tt,"2014-09-13 10:52:36");
 std::cout << t << std::endl;
 assert(!strcmp(t,kTime));
 assert(tt.tm_year == (2014-1900));
 assert(tt.tm_mon == 9-1);
 assert(tt.tm_mday == 13);
 assert(tt.tm_hour == 18);
 assert(tt.tm_min == 52);
 assert(tt.tm_sec == 36);
 
 return 0;
}
 
 
}

输出

?
1
2
3
4
Source DateTime: 2014-09-13T10:52:36Z
Dest DateTime: 2014-09-13 18:52:36
2014-09-13 18:52:36
2014-09-13 18:52:36

C++中获取UTC时间精确到微秒的实现代码

在日常开发过程中经常会使用到时间类函数的统计,其中获取1970年至今的UTC时间是比较常使用的,但是在windows下没有直接能够精确到微妙级的函数可用。本文提供方法正好可以解决这类需求问题。

下面先给出C++实现代码:

代码如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
#ifndef UTC_TIME_STAMP_H_
#define UTC_TIME_STAMP_H_
 
#include <windows.h>
#include <sys/timeb.h>
#include <time.h>
 
#if !defined(_WINSOCK2API_) && !defined(_WINSOCKAPI_)
struct timeval
{
long tv_sec;
long tv_usec;
};
#endif
 
static int gettimeofday(struct timeval* tv)
{
 union {
    long long ns100;
    FILETIME ft;
 } now;
 GetSystemTimeAsFileTime (&now.ft);
 tv->tv_usec = (long) ((now.ns100 / 10LL) % 1000000LL);
 tv->tv_sec = (long) ((now.ns100 - 116444736000000000LL) / 10000000LL);
 
 return (0);
}
//获取1970年至今UTC的微妙数
static time_t TimeConversion::GetUtcCaressing()
{
 timeval tv;
 gettimeofday(&tv);
 return ((time_t)tv.tv_sec*(time_t)1000000+tv.tv_usec);
}
#endif

接下来给出使用方法:

?
1
2
timeval tv;
gettimeofday(&tv);

或者直接调用:GetUtcCaressing();

UTC时间秒级UTC获取方法代码:

?
1
2
3
4
5
6
time_t timep;
struct tm *p;
time(&timep);
p=localtime(&timep);
timep = mktime(p);
printf("%d\n",timep);

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。

参考

原文链接:http://blog.csdn.net/infoworld/article/details/55190604