• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            Prayer

            在一般中尋求卓越
            posts - 1256, comments - 190, trackbacks - 0, articles - 0
              C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

            Linux下的時間相關(guān)

            Posted on 2009-02-04 18:07 Prayer 閱讀(624) 評論(0)  編輯 收藏 引用 所屬分類: LINUX/UNIX/AIX
                  遇到一個問題:程序調(diào)用usleep,發(fā)現(xiàn)要休眠的時間始終達不到效果。后來一查才發(fā)現(xiàn)問題是usleep()是微秒,不是之前理解的毫秒,難怪達不到效果。遙想當初為了避免sleep()秒級休眠的時間跨度較大,所以特意用了usleep(),在PHP中usleep是毫秒級的,導(dǎo)致自己先入為主,真值得檢討。

                   順便查閱一些linux下關(guān)于時間的資料和函數(shù),還挺有意思的:

            1、Linux下的時間
            1.1、Linux下的時間系統(tǒng)

                   UNIX及Linux的時間系統(tǒng)是由"新紀元時間"Epoch(傳說中的標志Unix時代開端的那個拂曉)開始計算起,單位為秒,Epoch則是指定為1970年一月一日凌晨零點零分零秒,格林威治時間。目前大部份的UNIX系統(tǒng)都是用32位來記錄時間,正值表示為1970以後,負值則表示1970年以前。我們可以很簡單地計算出其時間范圍: 2^31/86400(s) = 24855.13481(天) ~ 68.0958(年)

            1970+68.0958 = 2038.0958
            1970-68.0958 = 1901.9042

            時間范圍為[1901.9042,2038.0958]。

                  準確的時間為2038年一月十八日星期一晚上十點十四分七秒。那一刻,時間將會轉(zhuǎn)為負數(shù),變成1901年十二月十三日黑色星期五下午三點四十五分五十二秒,這就是所謂的UNIX 2038 BUG,或者您也可戲稱為Jason hatchet bug。在大部份的UNIX上,并沒有所謂Y2K問題,不過都有2038年問題

            1.2、Linux下與時間有關(guān)的數(shù)據(jù)結(jié)構(gòu)
            struct timeval {
                int tv_sec;
                int tv_usec;
            };
            其中tv_sec是由凌晨開始算起的秒數(shù),tv_usec則是微秒(10E-6 second)。

            struct timezone {
                int tv_minuteswest;
                int tv_dsttime;
            };
            tv_minuteswest是格林威治時間往西方的時差,tv_dsttime則是時間的修正方式。

            struct timespec
            {
                long int tv_sec;
                long int tv_nsec;
            };
            tv_nsec是nano second(10E-9 second)。

            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;
            };
            tm_sec表「秒」數(shù),在[0,61]之間,多出來的兩秒是用來處理跳秒問題用的。
            tm_min表「分」數(shù),在[0,59]之間。
            tm_hour表「時」數(shù),在[0,23]之間。
            tm_mday表「本月第幾日」,在[1,31]之間。
            tm_mon表「本年第幾月」,在[0,11]之間。
            tm_year要加1900表示那一年。
            tm_wday表「本第幾日」,在[0,6]之間。
            tm_yday表「本年第幾日」,在[0,365]之間,閏年有366日。
            tm_isdst表是否為「日光節(jié)約時間」。

            struct itimerval {
            struct timeval it_interval;
            struct timeval it_value;
            };
            it_interval成員表示間隔計數(shù)器的初始值,而it_value成員表示間隔計數(shù)器的當前值。

            2、獲得當前時間

            在所有的UNIX下,都有個time()的函數(shù)
            time_t time(time_t *t);
            這個函數(shù)會傳回從epoch開始計算起的秒數(shù),如果t是non-null,它將會把時間值填入t中。

            對某些需要較高精準度的需求,Linux提供了gettimeofday()。
            int gettimeofday(struct timeval * tv,struct timezone *tz);
            int settimeofday(const struct timeval * tv,const struct timezone *tz);

            struct tm格式時間函數(shù)

            struct tm * gmtime(const time_t * t);
            轉(zhuǎn)換成格林威治時間。有時稱為GMT或UTC。

            struct tm * localtime(const time_t *t);
            轉(zhuǎn)換成本地時間。它可以透過修改TZ環(huán)境變數(shù)來在一臺機器中,不同使用者表示不同時間。

            time_t mktime(struct tm *tp);
            轉(zhuǎn)換tm成為time_t格式,使用本地時間。

            tme_t timegm(strut tm *tp);
            轉(zhuǎn)換tm成為time_t格式,使用UTC時間。

            double difftime(time_t t2,time_t t1);
            計算秒差。

            文字時間格式函數(shù)
            char * asctime(struct tm *tp);
            char * ctime(struct tm *tp);
            這兩個函數(shù)都轉(zhuǎn)換時間格式為標準UNIX時間格式。
            Mon May 3 08:23:35 1999

            ctime一率使用當?shù)貢r間,asctime則用tm結(jié)構(gòu)內(nèi)的timezone資訊來表示。
            size_t strftime(char *str,size_t max,char *fmt,struct tm *tp);
            strftime有點像sprintf,其格式由fmt來指定。

            %a : 本第幾天名稱,縮寫。
            %A : 本第幾天名稱,全稱。
            %b : 月份名稱,縮寫。
            %B : 月份名稱,全稱。
            %c : 與ctime/asctime格式相同。
            %d : 本月第幾日名稱,由零算起。
            %H : 當天第幾個小時,24小時制,由零算起。
            %I : 當天第幾個小時,12小時制,由零算起。
            %j : 當年第幾天,由零算起。
            %m : 當年第幾月,由零算起。
            %M : 該小時的第幾分,由零算起。
            %p : AM或PM。
            %S : 該分鐘的第幾秒,由零算起。
            %U : 當年第幾,由第一個日開始計算。
            %W : 當年第幾,由第一個一開始計算。
            %w : 當?shù)趲兹眨闪闼闫稹?
            %x : 當?shù)厝掌凇?
            %X : 當?shù)貢r間。
            %y : 兩位數(shù)的年份。
            %Y : 四位數(shù)的年份。
            %Z : 時區(qū)名稱的縮寫。
            %% : %符號。

            char * strptime(char *s,char *fmt,struct tm *tp);
            如同scanf一樣,解譯字串成為tm格式。

            %h : 與%b及%B同。
            %c : 讀取%x及%X格式。
            %C : 讀取%C格式。
            %e : 與%d同。
            %D : 讀取%m/%d/%y格式。
            %k : 與%H同。
            %l : 與%I同。
            %r : 讀取"%I:%M:%S %p"格式。
            %R : 讀取"%H:%M"格式。
            %T : 讀取"%H:%M:%S"格式。
            %y : 讀取兩位數(shù)年份。
            %Y : 讀取四位數(shù)年份。

                下面舉一個小例子,說明如何獲得系統(tǒng)當前時間:
                time_t now;
                struct tm *timenow;
                char strtemp[255];

                time(&now);
                timenow = localtime(&now);
                printf("recent time is : %s \n", asctime(timenow));


            3、延時

            延時可以采用如下函數(shù):
            unsigned int sleep(unsigned int seconds);
            sleep()會使目前程式陷入「冬眠」seconds秒,除非收到「不可抵」的信號。
            如果sleep()沒睡飽,它將會返回還需要補眠的時間,否則一般返回零。

            void usleep(unsigned long usec);
            usleep與sleep()類同,不同之處在於秒的單位為10E-6秒。

            int select(0,NULL,NULL,NULL,struct timeval *tv);
            可以利用select的實作sleep()的功能,它將不會等待任何事件發(fā)生。

            int nanosleep(struct timespec *req,struct timespec *rem);
            nanosleep會沉睡req所指定的時間,若rem為non-null,而且沒睡飽,將會把要補眠的時間放在rem上。

            4、定時器

            4.1、alarm
            如果不要求很精確的話,用 alarm() 和 signal() 就夠了
            unsigned int alarm(unsigned int seconds)
            專門為SIGALRM信號而設(shè),在指定的時間seconds秒后,將向進程本身發(fā)送SIGALRM信號,又稱為鬧鐘時間。進程調(diào)用alarm后,任何以前的alarm()調(diào)用都將無效。如果參數(shù)seconds為零,那么進程內(nèi)將不再包含任何鬧鐘時間。如果調(diào)用alarm()前,進程中已經(jīng)設(shè)置了鬧鐘時間,則返回上一個鬧鐘時間的剩余時間,否則返回0。

            示例:
            #include <stdio.h>
            #include <unistd.h>
            #include <signal.h>

            void sigalrm_fn(int sig)
            {
                     /* Do something */
                     printf("alarm!\n");

                     alarm(2);
                     return;
            }

            int main(void)
            {
                     signal(SIGALRM, sigalrm_fn);
                     alarm(2);

                     /* Do someting */
                     while(1) pause();
            }

            4.2、setitimer
            int setitimer(int which, const struct itimerval *value, struct itimerval *ovalue));
            setitimer()比alarm功能強大,支持3種類型的定時器:
            ITIMER_REAL :   以系統(tǒng)真實的時間來計算,它送出SIGALRM信號。  
            ITIMER_VIRTUAL :   以該行程真正有執(zhí)行的時間來計算,它送出SIGVTALRM信號。  
            ITIMER_PROF :   以行程真正有執(zhí)行及在核心中所費的時間來計算,它送出SIGPROF信號。  
            Setitimer()第一個參數(shù)which指定定時器類型(上面三種之一);第二個參數(shù)是結(jié)構(gòu)itimerval的一個實例;第三個參數(shù)可不做處理。

            Setitimer()調(diào)用成功返回0,否則返回-1。
                 下面是關(guān)于setitimer調(diào)用的一個簡單示范,在該例子中,每隔一秒發(fā)出一個SIGALRM,每隔0.5秒發(fā)出一個SIGVTALRM信號::
            #include <stdio.h>
            #include <stdlib.h>
            #include <unistd.h>
            #include <signal.h>
            #include <time.h>
            #include <sys/time.h>

            int sec;
            void sigroutine(int signo){
                 switch (signo){
                 case SIGALRM:
                     printf("Catch a signal -- SIGALRM \n");
                     signal(SIGALRM, sigroutine);
                     break;
                 case SIGVTALRM:
                     printf("Catch a signal -- SIGVTALRM \n");
                     signal(SIGVTALRM, sigroutine);
                     break;
                 }
                 return;
            }

            int main()
            {
                 struct itimerval value, ovalue, value2;
                
                 sec = 5;
                 printf("process id is %d ", getpid());
                 signal(SIGALRM, sigroutine);
                 signal(SIGVTALRM, sigroutine);
                 value.it_value.tv_sec = 1;
                 value.it_value.tv_usec = 0;
                 value.it_interval.tv_sec = 1;
                 value.it_interval.tv_usec = 0;
                 setitimer(ITIMER_REAL, &value, &ovalue);

                 value2.it_value.tv_sec = 0;
                 value2.it_value.tv_usec = 500000;
                 value2.it_interval.tv_sec = 0;
                 value2.it_interval.tv_usec = 500000;
                 setitimer(ITIMER_VIRTUAL, &value2, &ovalue);
            }

            該例子的屏幕拷貝如下:
            localhost:~$ ./timer_test
            process id is 579
            Catch a signal – SIGVTALRM
            Catch a signal – SIGALRM
            Catch a signal – SIGVTALRM
            Catch a signal – SIGVTALRM
            Catch a signal – SIGALRM
            Catch a signal –GVTALRM
                 注意:Linux信號機制基本上是從Unix系統(tǒng)中繼承過來的。早期Unix系統(tǒng)中的信號機制比較簡單和原始,后來在實踐中暴露出一些問題,因此,把那些建立在早期機制上的信號叫做"不可靠信號",信號值小于SIGRTMIN(Red hat 7.2中,SIGRTMIN=32,SIGRTMAX=63)的信號都是不可靠信號。這就是"不可靠信號"的來源。它的主要問題是:進程每次處理信號后,就將對信號的響應(yīng)設(shè)置為默認動作。在某些情況下,將導(dǎo)致對信號的錯誤處理;因此,用戶如果不希望這樣的操作,那么就要在信號處理函數(shù)結(jié)尾再一次調(diào)用signal(),重新安裝該信號。 
            99久久国产亚洲综合精品| 99久久夜色精品国产网站 | 中文无码久久精品| 精品国产青草久久久久福利| 久久久久国产一级毛片高清版| 久久夜色精品国产噜噜噜亚洲AV| 久久香综合精品久久伊人| 热99RE久久精品这里都是精品免费 | 99热成人精品免费久久| 精品久久久久久| 国产精品久久久久9999| 69SEX久久精品国产麻豆| 91精品国产色综合久久| 亚洲国产二区三区久久| 国产亚洲美女精品久久久| 国产成人精品久久| 综合久久给合久久狠狠狠97色 | 91精品国产色综合久久| 色综合久久综合网观看| 久久精品国产一区二区三区不卡 | 国产AV影片久久久久久| 亚洲国产精品嫩草影院久久| 久久AV无码精品人妻糸列| 99蜜桃臀久久久欧美精品网站| 精品久久久久中文字幕日本| 中文字幕久久欲求不满| 少妇久久久久久被弄到高潮| 亚洲精品无码久久久久| 99久久综合狠狠综合久久| 区久久AAA片69亚洲| 99久久99久久久精品齐齐| 久久精品亚洲男人的天堂| 亚洲AV日韩AV永久无码久久| 一本久久久久久久| 亚洲AV日韩精品久久久久久久 | 狠狠人妻久久久久久综合| 99久久无色码中文字幕人妻 | 久久精品成人| 久久夜色精品国产噜噜麻豆 | 国产欧美久久久精品影院| 亚洲天堂久久精品|