C/C++標(biāo)準(zhǔn)庫(kù)之轉(zhuǎn)換UTC時(shí)間到local本地時(shí)間詳解
前言
UTC 時(shí)間DateTime.UtcNow 和 系統(tǒng)本地時(shí)間 DateTime.Now 相差8個(gè)時(shí)區(qū) ,美國(guó)本地時(shí)間和北京時(shí)間相差15個(gè)時(shí)區(qū): 美國(guó),而一般使用UTC時(shí)間方便統(tǒng)一各地區(qū)時(shí)間差異。
場(chǎng)景
1.如果有面向全球用戶的網(wǎng)站, 一般在存儲(chǔ)時(shí)間數(shù)據(jù)時(shí)存儲(chǔ)的是UTC格式的時(shí)間, 這樣時(shí)間是統(tǒng)一的, 并可以根據(jù)當(dāng)?shù)貢r(shí)區(qū)來(lái)進(jìn)行準(zhǔn)確的轉(zhuǎn)換.
2.存儲(chǔ)本地時(shí)間的問(wèn)題就在于如果換了時(shí)區(qū), 那么顯示的時(shí)間并不正確. 所以我們存儲(chǔ)時(shí)間時(shí)最好還是存儲(chǔ)UTC時(shí)間,便于正確的轉(zhuǎn)換.
說(shuō)明
1.C/C++標(biāo)準(zhǔn)庫(kù)提供了標(biāo)準(zhǔn)函數(shù)可以轉(zhuǎn)換, 不需要借助Win32 API.
例子
// test_datetime_format.cpp : 定義控制臺(tái)應(yīng)用程序的入口點(diǎn)。
//
#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;
}
}
輸出
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時(shí)間精確到微秒的實(shí)現(xiàn)代碼
在日常開(kāi)發(fā)過(guò)程中經(jīng)常會(huì)使用到時(shí)間類(lèi)函數(shù)的統(tǒng)計(jì),其中獲取1970年至今的UTC時(shí)間是比較常使用的,但是在windows下沒(méi)有直接能夠精確到微妙級(jí)的函數(shù)可用。本文提供方法正好可以解決這類(lèi)需求問(wèn)題。
下面先給出C++實(shí)現(xiàn)代碼:
代碼如下:
#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的微妙數(shù)
static time_t TimeConversion::GetUtcCaressing()
{
timeval tv;
gettimeofday(&tv);
return ((time_t)tv.tv_sec*(time_t)1000000+tv.tv_usec);
}
#endif
接下來(lái)給出使用方法:
timeval tv; gettimeofday(&tv);
或者直接調(diào)用:GetUtcCaressing();
UTC時(shí)間秒級(jí)UTC獲取方法代碼:
time_t timep;
struct tm *p;
time(&timep);
p=localtime(&timep);
timep = mktime(p);
printf("%d\n",timep);
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
參考
相關(guān)文章
C/C++中時(shí)間庫(kù)函數(shù)的使用詳解
這篇文章主要為大家詳細(xì)介紹了C/C++中的時(shí)間相關(guān)知識(shí)總結(jié),例如時(shí)間庫(kù)函數(shù)的使用以及獲取本地時(shí)間的不同方法,文中的示例代碼講解詳細(xì),需要的可以參考一下2022-11-11
詳解C++中的函數(shù)調(diào)用和下標(biāo)以及成員訪問(wèn)運(yùn)算符的重載
這篇文章主要介紹了詳解C++中的函數(shù)調(diào)用和下標(biāo)以及成員訪問(wèn)運(yùn)算符,講到了這些二元運(yùn)算符使用的語(yǔ)法及重載,需要的朋友可以參考下2016-01-01
C++中實(shí)現(xiàn)保存數(shù)據(jù)到CSV文件
這篇文章主要介紹了C++中實(shí)現(xiàn)保存數(shù)據(jù)到CSV文件方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08

