首頁 > 軟體

C++時間函數整理詳解

2022-10-30 14:01:34

一、 時間概念

格林威治時間GMT(Greenwich Mean Time)

格林威治皇家天文臺為了海上霸權的擴張計劃,在十七世紀就開始進行天體觀測。為了天文觀測,選擇了穿過英國倫敦格林威治天文臺子午儀中心的一條經線作為零度參考線,這條線,簡稱格林威治子午線。

1884年10月召開了一個國際子午線會議,該會議將格林威治子午線設定為本初子午線,並將格林威治平時 (GMT, Greenwich Mean Time) 作為世界時間標準(UT, Universal Time)。由此也確定了全球24小時自然時區的劃分,所有時區都以和 GMT 之間的偏移量做為參考。

1972年之前,格林威治時間(GMT)一直是世界時間的標準。1972年之後,GMT 不再是一個時間標準了。

XP系統中,預設時間格式是GMT。

目前UTC與GMT 相差為0.9秒,故二者可以基本視為一致。

原子時間

1967年,人們利用銫原子振盪週期極為規律的特性,研製出了高精度的原子時鐘,將銫原子能級躍遷輻射9192631770周所經歷的時間定為1s。現在用的時間就是1971年10月定義的國際原子時,是通過世界上大約200多臺原子鐘進行對比後,再由國際度量衡局時間所進行資料處理,得出的統一的原子時,簡稱TAI。

世界協調時

又稱世界統一時間、世界標準時間。UTC是現在全球通用的時間標準,全球各地都同意將各自的時間進行同步協調。UTC 時間是經過平均太陽時(以格林威治時間GMT為準)、地軸運動修正後的新時標以及以秒為單位的國際原子時所綜合精算而成。

UTC 構成:

原子時間(TAI):

結合了全球400個所有的原子鐘而得到的時間,它決定了我們每個人的鐘表中,時間流動的速度。

世界時間(UT, Universal Time):

也稱天文時間,或太陽時,他的依據是地球的自轉,我們用它來確定多少原子時,對應於一個地球日的時間長度。

格式: YYYY-MM-DDThh:mm:ssZ

協調世界時不與任何地區位置相關,也不代表此刻某地的時間,所以在說明某地時間時要加上時區。也就是說GMT並不等於UTC,而是等於UTC+0,只是格林尼治剛好在0時區上。GMT = UTC+0。

本地時間

在日常生活中所使用的時間我們通常稱之為本地時間。這個時間等於我們所在(或者所使用)時區內的當地時間,它由與世界標準時間(UTC)之間的偏移量來定義。這個偏移量可以表示為 UTC- 或 UTC+,後面接上偏移的小時和分鐘數。

GMT是前世界標準時,UTC是現世界標準時。

UTC 比 GMT更精準,以原子時計時,適應現代社會的精確計時。

但在不需要精確到秒的情況下,二者可以視為等同。

每年格林尼治天文臺會發調時資訊,基於UTC。

二、Linux-c時間的儲存方式

1. time_t

一個整型,儲存從1970-1-1 00:00:00年到現在UTC+0經過了多少秒,進一步的,struct timeval可精確到微秒。

2. 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*/  1+p->tm_mon;
	    int tm_year;  /*年, 從1900至今已經多少年*/  1900+ p->tm_year;
	    int tm_wday; /*星期,一週中的第幾天, 從星期日算起,0-6*/
	    int tm_yday; /*從今年1月1日到目前的天數,範圍0-365*/
	    int tm_isdst; /*日光節約時間的旗標*/
	};
特別注意,年份是從1900年起至今多少年,而不是直接儲存如2022年。月份從0開始的,0表示一月,星期也是從0開始的, 0表示星期日,1表示星期一。

三、 常用函數

#include <time.h>

#include <sys/time.h>

time_t time(time_t* t);

取得從1970-1-1 00:00:00至今(UTC+0)的秒數,注意並不是至本地時間的秒數。該時間戳為GMT時間,即時區為0。

time()總是返回的是當前格林威治時間,不論系統/程式採用的哪個時區。所以使用time_t時候,讓time_t上的儲存值總是描述格林威治時間。

    time_t tm = time(nullptr);//取得從1970-1-1 00:00:00至今(UTC+0)的秒數
    //或
    time(&tm0);

struct tm* gmtime(const time_t* timep);

將time_t表示的時間戳轉換為沒有經過時區轉換的UTC時間,是一個struct tm結構指標。

    struct tm *tmStamp = nullptr;
    time_t tmSec = time(nullptr); 
    tmStamp = gmtime(&tmSec);   //返回沒有經過時區轉換的UTC時間 struct tm結構

struct tm* localtime(const time_t* timep);

將time_t表示的時間戳轉換換成經過時區轉換的時間。使用time_t格林威治時間+時區偏差,生成tm結構

    struct tm *tmStamp = nullptr;
    time_t tmSec;
    time(&tmSec); //取得從1970年1月1日至今的秒數
    tmStamp = localtime(&tmSec); //返回經過時區轉換的時間,***注意本函數可能會修改時間秒的值***

time_t mktime(struct tm* timeptr);

將struct tm 結構的時間轉換為從1970-1-1年至今的秒數。

mktime考慮了時區,輸入的值總是要求【localtime-tm】結構-當前時區時間,輸出值格林威治時間;

char asctime(const struct tm timeptr);

將struct tm結構中的資訊轉換為真實世界的時間(不經過時區轉換的UTC時間),以字串的形式顯示。

    time_t timeSec;
    time(&timeSec); /*獲取time_t型別的當前時間*/
    /*用gmtime將time_t型別的時間轉換為struct tm型別的時間,按沒有經過時區轉換的UTC時間
      然後再用asctime轉換為我們常見的格式 ,形式:Mon Oct 24 11:41:17 2022
    */
    printf("%s", asctime(gmtime(&timeSec))); //沒有經過時區轉換的UTC時間

char ctime(const time_t timep);

將 time_t時間秒錶示的時間轉換為真實世界的時間(經時區轉換的UTC時間),以字串顯示。

ctime考慮了時區,輸入值要求time_t是格林威治時間,輸出來的值總是用來描述當地時間。

// 形式:Mon Oct 24 11:41:17 2022
``
## 7. gettimeofday(&tmval,&zone); // for linux 
返回當前距離1970年的秒數和微妙數,後面的tz是時區,一般不用傳NULL。
```c
	struct timezone zone;
    struct timeval tmval;
    gettimeofday(&tmval,&zone); // for linux

double difftime(time_t time1, time_t time2);

返回兩個時間秒相差的秒數。

ctime返回的是靜態變數地址;更要注意gmtime與localtime返回的靜態變數地址是同一個,後呼叫的會覆蓋上次呼叫的值;

tm結構儲存值有時用來描述格林威治時間gmtime,有時用來描述當地時間localtime-當前時區時間。

四、 時間格式化

size_t strftime(char *str, size_t count, const char *format, const struct tm *tm)

函數原型
#include <time.h>
size_t strftime(char *str, size_t count, const char *format, const struct tm *tm);
引數說明
  str, 表示返回的時間字串
  count, 要寫入的位元組的最大數量
  format, 格式字串由零個或多個轉換符和普通字元(除%)
  tm, 輸入時間
返回值
  如果包含終止的空字元在內的結果字元的總數不大於count,則函數strftime返回字元數,這些字元被放到s指向的陣列中但不包含終止的空字元。否則,函數返回零,且陣列的內容不確定。

一個常規用法

	char* format = "%Y-%m-%d %H:%M:%S";
    char strTime[100];
    strftime(strTime, sizeof(strTime), format, tmTime);//2022-10-07 20:46:01

五、計時器-時間段

#include <time.h>
#include <iostream>
using namespace std;
clock_t start = clock();
// do something...
clock_t end   = clock();
cout << "花費了" << (double)(end - start) / CLOCKS_PER_SEC << "秒" << endl;

可精確到毫秒

六、chrono

C++ 11 標準庫引入了chrono庫。利用該庫可以做時間運算和換算。

一個效能統計用法

#include <chrono>   
using namespace std;
using namespace chrono;
auto start = system_clock::now();
// do something...
auto end   = system_clock::now();
auto duration = duration_cast<microseconds>(end - start);
cout <<  "花費了" 
     << double(duration.count()) * microseconds::period::num / microseconds::period::den 
     << "秒" << endl;
//精確到微妙

到此這篇關於C++時間函數整理詳解的文章就介紹到這了,更多相關C++時間函數內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com