Linux/Unix time時間戳的處理轉換函數linux
linux下的時間函數
咱們在編程中可能會常常用到時間,好比取得系統的時間(獲取系統的年、月、日、時、分、秒,星期等),或者是隔一段時間去作某事,那麼咱們就用到一些時間函數。
linux下存儲時間常見的有兩種存儲方式:
一個是從1970年到如今通過了多少秒;
一個是用一個結構來分別存儲年月日時分秒的。
time_t 這種類型就是用來存儲從1970年到如今通過了多少秒,要想更精確一點,能夠用結構struct timeval,它精確到微妙。編程
1 struct timeval 2 { 3 long tv_sec; /*秒*/ 4 long tv_usec; /*微秒*/ 5 };
而直接存儲年月日的是一個結構:數組
1 struct tm 2 { 3 int tm_sec; /*秒,正常範圍0-59, 但容許至61*/ 4 int tm_min; /*分鐘,0-59*/ 5 int tm_hour; /*小時, 0-23*/ 6 int tm_mday; /*日,即一個月中的第幾天,1-31*/ 7 int tm_mon; /*月, 從一月算起,0-11*/ 8 int tm_year; /*年, 從1900至今已經多少年*/ 9 int tm_wday; /*星期,一週中的第幾天, 從星期日算起,0-6*/ 10 int tm_yday; /*從今年1月1日到目前的天數,範圍0-365*/ 11 int tm_isdst; /*日光節約時間的旗標*/ 12 };
須要特別注意的是,年份是從1900年起至今多少年,而不是直接存儲如2008年,月份從0開始的,0表示一月,星期也是從0開始的, 0表示星期日,1表示星期一。函數
下面介紹一下咱們經常使用的時間函數:spa
1 #include <time.h> 2 3 char *asctime(const struct tm* timeptr); //將結構中的信息轉換爲真實世界的時間,以字符串的形式顯示; 4 5 char *ctime(const time_t *timep); //將timep轉換爲真是世界的時間,以字符串顯示,它和asctime不一樣就在於傳入的參數形式不同; 6 7 double difftime(time_t time1, time_t time2); //返回兩個時間相差的秒數; 8 9 int gettimeofday(struct timeval *tv, struct timezone *tz); //返回當前距離1970年的秒數和微妙數,後面的tz是時區,通常不用; 10 11 struct tm* gmtime(const time_t *timep); //將time_t表示的時間轉換爲沒有通過時區轉換的UTC時間,是一個struct tm結構指針; 12 13 stuct tm* localtime(const time_t *timep); //和gmtime相似,可是它是通過時區轉換的時間。 14 15 time_t mktime(struct tm* timeptr); //將struct tm 結構的時間轉換爲從1970年至今的秒數; 16 17 time_t time(time_t *t); //取得從1970年1月1日至今的秒數。
上面是簡單的介紹,下面經過實戰來看看這些函數的用法:
下載: gettime1.c指針
1 /*gettime1.c*/ 2 #include <time.h> 3 int main() 4 { 5 time_t timep; 6 time(&timep); /*獲取time_t類型的當前時間*/ 7 /*用gmtime將time_t類型的時間轉換爲struct tm類型的時間按, 8 而後再用asctime轉換爲咱們常見的格式 Fri Jan 11 17:25:24 2008 9 */ 10 printf("%s", asctime(gmtime(&timep))); 11 return 0; 12 }
編譯並運行:code
$gcc -o gettime1 gettime1.c $./gettime1 Fri Jan 11 17:04:08 2008
下面是直接把time_t類型的轉換爲咱們常見的格式:
下載: gettime2.corm
13 /* gettime2.c*/ 14 #include <time.h> 15 int main() 16 { 17 time_t timep; 18 time(&timep); /*獲取time_t類型當前時間*/ 19 /*轉換爲常見的字符串:Fri Jan 11 17:04:08 2008*/ 20 printf("%s", ctime(&timep)); 21 return 0; 22 } 編譯並運行: $gcc -o gettime2 gettime2.c $./gettime2 Sat Jan 12 01:25:29 2008
我看了一本書上面說的這兩個例子若是前後執行的話,兩個的結果除了秒上有差異以外(執行程序須要時間),應該是同樣的,但是我這裏執行卻發現差了很長時間按,一個是週五,一個是週六,後來我用 date 命令執行了一遍
$date
六 1月 12 01:25:19 CST 2008
我發現date和gettime2比較一致, 我估計可能gettime1並無通過時區的轉換,它們是有差異的。
下載: gettime3.cblog
23 /*gettime3.c */ 24 #include <time.h> 25 int main() 26 { 27 char *wday[] = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"}; 28 time_t timep; 29 struct tm *p; 30 time(&timep); /*得到time_t結構的時間,UTC時間*/ 31 p = gmtime(&timep); /*轉換爲struct tm結構的UTC時間*/ 32 printf("%d/%d/%d ", 1900 + p->tm_year, 1 + p->tm_mon, p->tm_mday); 33 printf("%s %d:%d:%d/n", wday[p->tm_wday], p->tm_hour, 34 p->tm_min, p->tm_sec); 35 return 0; 36 } 編譯並運行: $gcc -o gettime3 gettime3.c $./gettime3 2008/1/11 Fri 17:42:54
從這個時間結果上來看,它和gettime1保持一致。
下載: gettime4.c進程
37 /*gettime4.c*/ 38 #include <time.h> 39 int main() 40 { 41 char *wday[] = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"}; 42 time_t timep; 43 struct tm *p; 44 time(&timep); /*得到time_t結構的時間,UTC時間*/ 45 p = localtime(&timep); /*轉換爲struct tm結構的當地時間*/ 46 printf("%d/%d/%d ", 1900 + p->tm_year, 1 + p->tm_mon, p->tm_mday); 47 printf("%s %d:%d:%d/n", wday[p->tm_wday], p->tm_hour, p->tm_min, p->tm_sec); 48 return 0; 49 } 編譯並運行: $gcc -o gettime4 gettime4.c $./gettime4 2008/1/12 Sat 1:49:29
從上面的結果咱們能夠這樣說:
time, gmtime, asctime 所表示的時間都是UTC時間,只是數據類型不同,而
localtime, ctime 所表示的時間都是通過時區轉換後的時間,它和你用系統命令date所表示的CST時間應該保持一致。
下載: gettime5.c
50 /*gettime5.c*/ 51 #include <time.h> 52 int main() 53 { 54 time_t timep; 55 struct tm *p; 56 time(&timep); /*當前time_t類型UTC時間*/ 57 printf("time():%d/n",timep); 58 p = localtime(&timep); /*轉換爲本地的tm結構的時間按*/ 59 timep = mktime(p); /*從新轉換爲time_t類型的UTC時間,這裏有一個時區的轉換*/ 60 printf("time()->localtime()->mktime(): %d/n", timep); 61 return 0; 62 } 編譯並運行: $gcc -o gettime5 gettime5.c $./gettime5 time():1200074913 time()->localtime()->mktime(): 1200074913
這裏面把UTC時間按轉換爲本地時間,而後再把本地時間轉換爲UTC時間,它們轉換的結果保持一致。
下載: gettime6.c
63 /*gettime6.c */ 64 #include <time.h> 65 int main() 66 { 67 time_t timep; 68 struct tm *p; 69 time(&timep); /*獲得time_t類型的UTC時間*/ 70 printf("time():%d/n",timep); 71 p = gmtime(&timep); /*獲得tm結構的UTC時間*/ 72 timep = mktime(p); /*轉換,這裏會有時區的轉換*/ 73 printf("time()->gmtime()->mktime(): %d/n", timep); 74 return 0; 75 } 編譯並運行: $gcc -o gettime6 gettime6.c $./gettime6 time():1200075192 time()->gmtime()->mktime(): 1200046392
從這裏面咱們能夠看出,轉換後時間不一致了,計算一下,整整差了8個小時( (1200075192-1200046392)/3600 = 8 ),說明mktime會把本地時間轉換爲UTC時間,這裏面原本就是UTC時間,因而再弄個時區轉換,結果差了8個小時,用的時候應該注意。
函數分類:
1. 設置時間: settimeofday, tzset
2. 獲取時間: time, ftime, gettimeofday
3. 時間格式轉換: mktime, strftime; gmtime, localtime; asctime, ctime
4. 其餘: clock, difftime
asctime: 將時間和日期以字符串格式表示
頭文件: time.h
函數定義: char *asctime(const struct tm *timeptr);
說明: asctime()將函數timeptr所指的tm結構中的信息轉換成現實世界所使用的時間日期表示方法, 而後將結果以字符串形態返回. 此函數已經由時區轉換成當地時間, 返回的字符串格式爲: "Wed Jun 30 21:49:08 1993/n"
ctime: 將時間和日期以字符串格式表示
頭文件: time.h
函數定義: char *ctime(const time_t *timep);
說明: ctime()同asctime()函數, 只是輸入參數爲time_t.
應用舉例:
#include <stdio.h> #include <time.h> int main(void) { time_t timep; time(&timep); printf("%s", ctime(&timep)); printf("%s", asctime(gmtime(&timep))); return 0; } 運行結果: Sun Dec 14 15:30:11 2008 Sun Dec 14 15:30:11 2008
clock: 取得進程佔用CPU的大約時間
頭文件: time.h
函數定義: clock_t clock(void);
說明: clock()用來返回進程所佔用CPU的大約時間.
difftime: 計算時間差距
頭文件: time.h
函數定義: double difftime(time_t time1, time_t time0);
說明: difftime()用來計算參數time1-time0, 結果以double型精確值返回. 兩個參數的時間都是以1970年1月1日0時0分0秒算起的UTC時間.
ftime: 取得目前的時間和日期
頭文件: sys/timeb.h
函數定義: int ftime(struct timeb *tp);
說明: ftime()將日前時間日期由參數tp所指的結構輸出. tp結構定義以下:
struct timeb{
/* 爲從1970年1月1日至今的秒數 */
time_t time;
/* 爲千分之一秒 */
unsigned short millitm;
/* 爲目前時區和Greenwich相差的時間, 單位爲單位 */
short timezone;
/* 爲日光節約時間的修正狀態, 若非0爲啓用日光節約時間的修正 */
short dstflag;
};
不管成功仍是失敗都返回0.
應用舉例:
#include <stdio.h> #include <sys/timeb.h> int main(void) { struct timeb tp; ftime(&tp); printf("time: %d/n", tp.time); printf("millitm: %d/n", tp.millitm); printf("timezone: %d/n", tp.timezone); printf("dstflag: %d/n", tp.dstflag); return 0; } 運行結果: time: 1229271908 millitm: 716 timezone: -480 dstflag: 0
gettimeofday: 取得目前的時間
頭文件: sys/time.h unist.d
函數定義: int gettimeofday(struct timeval *tv, struct timezone *tz);
說明: gettimeofday()會把目前的時間用tv所指的結構返回, 當地時區的信息則放到tz所指的結構中. 成功則返回0, 失敗返回-1, 錯誤代碼存於errno. EFAULT是指針tv和tz所指的內存空間超出存取權限.
timeval結構定義爲:
struct timeval{ /* 爲從1970年1月1日至今的秒數 */ long tv_sec; /* 微秒 */ long tv_usec; }; timezone 結構定義爲: struct timezone{ /* 和Greenwich時間差了多少分鐘 */ int tz_minuteswest; /*日光節約時間的狀態*/ int tz_dsttime; }; 上述兩個結構都定義在/usr/include/sys/time.h, tz_dsttime所表明的狀態以下: DST_NONE /*不使用*/ DST_USA /*美國*/ DST_AUST /*澳洲*/ DST_WET /*西歐*/ DST_MET /*中歐*/ DST_EET /*東歐*/ DST_CAN /*加拿大*/ DST_GB /*大不列顛*/ DST_RUM /*羅馬尼亞*/ DST_TUR /*土耳其*/ DST_AUSTALT /*澳洲(1986年之後)*/
gmtime: 將秒數轉換目前的時間和日期
頭文件: time.h
函數定義: struct tm *gmtime(const time_t *timep);
說明: gmtime()將參數timep所指的time_t結構中的信息轉換成現實世界所使用的時間日期表示方法, 而後將結果由結構tm返回.
結構tm的定義爲:
struct tm{ int tm_sec; int tm_min; int tm_hour; int tm_mday; int tm_mon; int tm_year; int tm_wday; int tm_yday; int tm_isdst; }; 參數說明: int tm_sec 表明目前秒數, 正常範圍爲0-59, 但容許至61秒 int tm_min 表明目前分數, 範圍0-59 int tm_hour 從午夜算起的時數, 範圍爲0-23 int tm_mday 目前月份的日數, 範圍01-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 日光節約時間的旗標
此函數返回的時間日期未經時區轉換, 是UTC時間.
localtime: 將秒數轉換當地目前的時間和日期
頭文件: time.h
函數定義: struct *localtime(const time_t *timep);
說明: localtime()將參數timep所指的time_t結構中的信息轉換成真實世界所使用的時間日期表示方法, 而後將結果由結構tm返回. 結構tm的定義請參考gmtime(). 此函數返回的時間日期已經轉換成當地時區.
mktime: 將時間結構數據轉換成通過的秒數
頭文件: time.h
函數定義: time_t mktime(struct tm *timeptr);
說明: mktime()用來將參數timeptr所指的tm結構數據轉換成從1970年1月1日0時0分0秒算起至今的UTC時間所通過的秒數. 返回通過的秒數.
settimeofday: 設置目前的時間
頭文件: sys/time.h unistd.h
函數定義: settimeofday()會把目前時間設成由tv所指的結構信息, 當地時區信息則設成tz所指的結構. 詳細的說明請參考gettimeofday(). 注意, 只有root權限才能使用此函數修改時間. 成功則返回0, 失敗返回-1, 錯誤代碼存於errno.
錯誤代碼:
EPERM 並不是由root權限調用settimeofday(), 權限不夠
EINVAL 時區或某個數據是不正確的, 沒法正確設置時間
strftime: 格式化日期和時間
頭文件: time.h
函數定義: size_t strftime(char *s, size_t max, const char *format, const struct tm *tm);
說明: strftime()會將參數tm的時間結構, 參照參數format所指定的字符串格式作轉換, 轉換後的字符串內容將複製到參數s所指的字符串數組中, 該字符串的最大長度爲參數max所控制.
下面是參數format的格式指令:
%a 當地星期日期的名稱縮寫, 如: Sun
%A 當地星期日期的名稱縮寫, 如: Sunday
%b 當地月份的縮寫
%B 當地月份的完整名稱
%c 當地適當的日期與時間表示法
%C 以year/100表示年份
%d 月裏的天數, 表示法爲01-31
%D 至關於"%m%d%y"格式
%e 月裏的天數, 表示法爲1-31
%h 當地月份的縮寫
%H 以24小時製表示小時數, 表示法爲00-23
%I 以12小時製表示小時數, 表示法爲01-12
%j 一年中的天數(001-366)
%k 以24小時製表示小時數, 表示法爲0-23
%l 以12小時製表示小時數, 表示法爲1-12
%m 月份(01-12)
%M 分鐘數(00-59)
%n 同/n
%p 顯示對應的AM或PM
%P 顯示對應的am或pm
%r 至關於使用"%I:%M:%S %p"格式
%R 至關於使用"%H:%M"格式
%s 從1970年1月1日0時0分0秒算起至今的UTC時間所通過的秒數
%S 秒數(00-59)
%t 同/t
%T 24小時時間表示, 至關於"%H:%M:%S"格式
%u 一星期中的星期日期, 範圍1-7, 星期一從1開始
%U 一年中的星期數(00-53), 一月第一個星期日開始爲01
%w 一星期中的星期日期, 範圍0-6, 星期日從0開始
%W 一年中的星期數(00-53), 一月第一個星期一開始爲01
%x 當地適當的日期表示
%X 當地適當的時間表示
%y 一世紀中的年份表示
%Y 完整的公元年份表示
%Z 使用的時區名稱
%% '%'符號
返回複製到參數s所指的字符串數組的總字符數, 不包括字符串結束符. 若是返回0, 表示未複製字符串到參數s內, 但不表示必定有錯誤發生.
附加說明: 環境變量TZ和TC_TIME會影響此函數結果.
應用舉例:
#include <stdio.h> #include <time.h> int main(void) { char *format[] = {"%I: %M: %S %p %m/%d %a", "%x %X %Y", NULL}; char buf[30]; int i; time_t clock; struct tm *tm; time(&clock); tm = localtime(&clock); for(i = 0; format[i] != NULL; i++) { strftime(buf, sizeof(buf), format[i], tm); printf("%s => %s/n", format[i], buf); } return 0; } 運行結果: %I: %M: %S %p %m/%d %a => 01: 46: 44 AM 12/15 Mon %x %X %Y => 12/15/08 01:46:44 2008
time: 取得目前的時間
頭文件: time.h
函數定義: time_t time(time_t *t);
說明: time()會返回從1970年1月1日從0時0分0秒算起的UTC時間到如今所通過的秒數. 若是t並不是空指針的話, 此函數也會將返回值存到t指針所指的內存. 成功則返回秒數, 失敗則返回(time_t-1)值, 錯誤緣由存於errno中.
tzset: 設置時區以供時間轉換 頭文件: time.h 函數定義: void tzset(void); extern char *tzname[2]; 說明: tzset()用來將環境變量TZ設給全局變量tzname, 也就是從環境變量取得目前當地的時區. 時間轉換函數會自動調用此函數. 若TZ爲設置, tzname會依照/etc/localtime找出最接近當地的時區. 若TZ爲NULL, 或是沒法判認, 則使用UTC時區. 此函數老是成功, 而且初始化tzname.