Unix/Linux環境C程式設計入門教程(28) 日期時間那些事兒

尹成發表於2014-07-20

記得這個專題第一篇我們寫過一個程式執行時間的程式,採用庫函式提供的clock()模擬做程式測試。本篇介紹的函式也是和時間相關,但是沒有clock的細緻,而是提供的系統時間和日期。

1.asctime() ctime() gettimeofday() gmtime() localtime() mktime() settimeofday() time()函式介紹


asctime(將時間和日期以字串格式表示)

相關函式

time,ctime,gmtime,localtime

表標頭檔案

#include<time.h>

定義函式

char * asctime(const struct tm * timeptr);

函式說明

asctime()將引數timeptr所指的tm結構中的資訊轉換成真實世界所使用的時間日期表示方法,然後將結果以字串形態返回。此函式已經由時區轉換成當地時間,字串格式為:"Wed Jun 30 21:49:08 1993\n"

返回值

若再呼叫相關的時間日期函式,此字串可能會被破壞。此函式與ctime不同處在於傳入的引數是不同的結構。

附加說明

返回一字串表示目前當地的時間日期。

範例

#include <time.h>
main()
{
time_t timep;
time (&timep);
printf("%s",asctime(gmtime(&timep)));
}

執行

Sat Oct 28 02:10:06 2000

   




ctime(將時間和日期以字串格式表示)

相關函式

time,asctime,gmtime,localtime

表標頭檔案

#include<time.h>

定義函式

char *ctime(const time_t *timep);

函式說明

ctime()將引數timep所指的time_t結構中的資訊轉換成真實世界所使用的時間日期表示方法,然後將結果以字串形態返回。此函式已經由時區轉換成當地時間,字串格式為"Wed Jun 30 21 :49 :08 1993\n"。若再呼叫相關的時間日期函式,此字串可能會被破壞。

返回值

返回一字串表示目前當地的時間日期。

範例

#include<time.h>
main()
{
time_t timep;
time (&timep);
printf("%s",ctime(&timep));
}

執行

Sat Oct 28 10 : 12 : 05 2000

   




gettimeofday(取得目前的時間)

相關函式

time,ctime,ftime,settimeofday

表標頭檔案

#include <sys/time.h>
#include <unistd.h>

定義函式

int gettimeofday ( struct timeval * tv , struct timezone * tz )

函式說明

gettimeofday()會把目前的時間有tv所指的結構返回,當地時區的資訊則放到tz所指的結構中。
timeval結構定義為:
struct timeval{
long tv_sec; /*秒*/
long tv_usec; /*微秒*/
};
timezone 結構定義為:
struct timezone{
int tz_minuteswest; /*和Greenwich 時間差了多少分鐘*/
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年以後)*/

返回值

成功則返回0,失敗返回-1,錯誤程式碼存於errno。附加說明EFAULT指標tv和tz所指的記憶體空間超出存取許可權。

範例

#include<sys/time.h>
#include<unistd.h>
main(){
struct timeval tv;
struct timezone tz;
gettimeofday (&tv , &tz);
printf("tv_sec; %d\n", tv,.tv_sec) ;
printf("tv_usec; %d\n",tv.tv_usec);
printf("tz_minuteswest; %d\n", tz.tz_minuteswest);
printf("tz_dsttime, %d\n",tz.tz_dsttime);
}

執行

tv_sec: 974857339
tv_usec:136996
tz_minuteswest:-540
tz_dsttime:0

   




gmtime(取得目前時間和日期)

相關函式

time,asctime,ctime,localtime

表標頭檔案

#include<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時間。

返回值

返回結構tm代表目前UTC 時間

範例

#include <time.h>
main(){
char *wday[]={"Sun","Mon","Tue","Wed","Thu","Fri","Sat"};
time_t timep;
struct tm *p;
time(&timep);
p=gmtime(&timep);
printf("%d%d%d",(1900+p->tm_year), (1+p->tm_mon),p->tm_mday);
printf("%s%d;%d;%d\n", wday[p->tm_wday], p->tm_hour, p->tm_min, p->tm_sec);
}

執行

2000/10/28 Sat 8:15:38

   




localtime(取得當地目前時間和日期)

相關函式

time, asctime, ctime, gmtime

表標頭檔案

#include<time.h>

定義函式

struct tm *localtime(const time_t * timep);

函式說明

localtime()將引數timep所指的time_t結構中的資訊轉換成真實世界所使用的時間日期表示方法,然後將結果由結構tm返回。結構tm的定義請參考gmtime()。此函式返回的時間日期已經轉換成當地時區。

返回值

返回結構tm代表目前的當地時間。

範例

#include<time.h>
main(){
char *wday[]={"Sun","Mon","Tue","Wed","Thu","Fri","Sat"};
time_t timep;
struct tm *p;
time(&timep);
p=localtime(&timep); /*取得當地時間*/
printf ("%d%d%d ", (1900+p->tm_year),( l+p->tm_mon), p->tm_mday);
printf("%s%d:%d:%d\n", wday[p->tm_wday],p->tm_hour, p->tm_min, p->tm_sec);
}

執行

2000/10/28 Sat 11:12:22

   




mktime(將時間結構資料轉換成經過的秒數)

相關函式

time,asctime,gmtime,localtime

表標頭檔案

#include<time.h>

定義函式

time_t mktime(strcut tm * timeptr);

函式說明

mktime()用來將引數timeptr所指的tm結構資料轉換成從公元1970年1月1日0時0分0 秒算起至今的UTC時間所經過的秒數。

返回值

返回經過的秒數。

範例

/* 用time()取得時間(秒數),利用localtime()
轉換成struct tm 再利用mktine()將struct tm轉換成原來的秒數*/
#include<time.h>
main()
{
time_t timep;
strcut tm *p;
time(&timep);
printf("time() : %d \n",timep);
p=localtime(&timep);
timep = mktime(p);
printf("time()->localtime()->mktime():%d\n",timep);
}

執行

time():974943297
time()->localtime()->mktime():974943297

   




settimeofday(設定目前時間)

相關函式

time,ctime,ftime,gettimeofday

表標頭檔案

#include<sys/time.h>
#include<unistd.h>

定義函式

int settimeofday ( const struct timeval *tv,const struct timezone *tz);

函式說明

settimeofday()會把目前時間設成由tv所指的結構資訊,當地時區資訊則設成tz所指的結構。詳細的說明請參考gettimeofday()。注意,只有root許可權才能使用此函式修改時間。

返回值

成功則返回0,失敗返回-1,錯誤程式碼存於errno。

錯誤程式碼

EPERM 並非由root許可權呼叫settimeofday(),許可權不夠。
EINVAL 時區或某個資料是不正確的,無法正確設定時間。

   




time(取得目前的時間)

相關函式

ctime,ftime,gettimeofday

表標頭檔案

#include<time.h>

定義函式

time_t time(time_t *t);

函式說明

此函式會返回從公元1970年1月1日的UTC時間從0時0分0秒算起到現在所經過的秒數。如果t 並非空指標的話,此函式也會將返回值存到t指標所指的記憶體。

返回值

成功則返回秒數,失敗則返回((time_t)-1)值,錯誤原因存於errno中。

範例

#include<time.h>
mian()
{
int seconds= time((time_t*)NULL);
printf("%d\n",seconds);
}

執行

9.73E+08

2.小試牛刀

下面我們來看看 關於時間日期我們能做的一些常規的操作。

流程設計:

  1. 正確的獲取到系統時間和日期
  1. 將獲取出來的時間減去兩分鐘
  2. 再獲取系統時間一次看是否設定有效

注意 root許可權執行可執行程式

原始碼:

#include <time.h>
#include <sys/time.h>
#include <stdio.h>
int main()
{
// 星期天 星期一 二 三 四 五 六
const char *wday[]={"Sun","Mon","Tue","Wed","Thu","Fri","Sat"};
time_t timep;
struct tm *p;
struct timeval tv;
struct timezone tz;
time(&timep);
p=gmtime(&timep);
printf("%d年 %d月 %d日\n",(1900+p->tm_year), (1+p->tm_mon),p->tm_mday);
//中國是東八區 所以我們在p->tm_hour + 8
printf("%s %d:%d:%d\n", wday[p->tm_wday], p->tm_hour+8, p->tm_min, p->tm_sec);
 
gettimeofday(&tv,&tz);
printf("tv_sec:%d\n", tv.tv_sec) ;
printf("tv_usec: %d\n",tv.tv_usec);
 
tv.tv_sec -=120;
printf("tv_sec:%d\n", tv.tv_sec) ;
 
settimeofday(&tv,&tz);
//#include<sys/time.h>
//#include<unistd.h>
 
// int settimeofday ( const struct timeval *tv,const struct timezone *tz);
/*timeval結構定義為:
struct timeval{
long tv_sec; 秒
long tv_usec; 微秒
 
timezone 結構定義為:
struct timezone{
int tz_minuteswest; 和Greenwich 時間差了多少分鐘
int tz_dsttime; 日光節約時間的狀態
};
*/
time_t timeq;
struct tm *q;
time(&timeq);
q=gmtime(&timeq);
printf("%d年 %d月 %d日\n",(1900+q->tm_year), (1+q->tm_mon),q->tm_mday);
//中國是東八區 所以我們在p->tm_hour + 8
printf("%s %d:%d:%d\n", wday[q->tm_wday], q->tm_hour+8, q->tm_min, q->tm_sec);
 
return 0;
}


3.各平臺的執行情況

在RHEL7上

在RHEL6上 我們不使用root使用者執行可執行程式

在Solaris11上

在MAC上

好了我們效果演示完了,請大家自行試驗RHEL6上用root使用者執行的itcast。

 

相關文章