• <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>

            桃源谷

            心靈的旅行

            人生就是一場(chǎng)旅行,不在乎旅行的目的地,在乎的是沿途的風(fēng)景和看風(fēng)景的心情 !
            posts - 32, comments - 42, trackbacks - 0, articles - 0
              C++博客 :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

            給文件鎖設(shè)定timeout

            Posted on 2007-11-26 14:52 lymons 閱讀(2472) 評(píng)論(0)  編輯 收藏 引用 所屬分類: CUnix/Linux
            From 2008精選
            Linux 文件鎖是建議鎖,也有人把它叫做記錄鎖,是通過(guò)系統(tǒng)調(diào)用fcntl(2)來(lái)實(shí)現(xiàn)的。
            這種鎖在鎖定文件時(shí)有兩種模式,分別是阻塞(block)和非阻塞模式。
            在編碼時(shí)比較常用的是有一種的非阻塞模式,也就是發(fā)現(xiàn)文件已經(jīng)被其他進(jìn)程
            鎖定時(shí),立即返回不予等待。而阻塞模式則正好與它相反,也就是一直等待直到
            其他進(jìn)程釋放文件鎖為止。
            注:關(guān)于詳細(xì)內(nèi)容請(qǐng)參看《Unix環(huán)境高級(jí)編程》

            不過(guò),有的時(shí)候也會(huì)用到阻塞模式的文件鎖,而且會(huì)要求不能被一直阻塞,等待
            了一定時(shí)間后應(yīng)返回。也就是說(shuō),想給阻塞版本的文件鎖加上一個(gè)超時(shí)時(shí)間(timeout)。

            通過(guò)man手冊(cè),fcntl(2)里面沒(méi)有關(guān)于在阻塞模式時(shí),設(shè)置超時(shí)時(shí)間的任何描述。
            但從man手冊(cè)里我們發(fā)現(xiàn),文件鎖在阻塞時(shí)會(huì)被信號(hào)(signal)中斷。所以我們就像
            可以利用設(shè)置信號(hào)軟中斷來(lái)實(shí)現(xiàn)一個(gè)自己版本的等待超時(shí)呢。

             1 #include <stdio.h>
             2 #include <stdlib.h>
             3 #include <unistd.h>
             4 #include <fcntl.h>
             5 #include <sys/types.h>
             6 #include <sys/stat.h>
             7 #include <signal.h>
             8 
             9 #define readw_lock(fd) \
            10     lock_reg((fd), F_SETLKW, F_RDLCK)
            11 #define writew_lock(fd) \
            12     lock_reg((fd), F_SETLKW, F_WRLCK)
            13 #define unlock(fd) \
            14     lock_reg((fd), F_SETLK, F_UNLCK)
            15 
            16 typedef int (*LW_FN)(char *fname);
            17 
            18 int
            19 lock_reg(int fd, int cmd, int type)
            20 {
            21     struct flock lock;
            22     lock.l_type = type;
            23     lock.l_start = 0;
            24     lock.l_whence = SEEK_SET;
            25     lock.l_len = 0;
            26 
            27     return fcntl(fd, cmd, &lock);
            28 }
            29 
            30 void hander(int signo)
            31 {
            32     // do nothing
            33     return;
            34 }
            35 
            36 int lockw(char *fname, LW_FN fn, int timeout)
            37 {
            38     int ret = 0;
            39     int fd;
            40     struct sigaction act, oact;
            41 
            42     if ((fd = open(fname,  O_CREAT | O_RDWR, 0666)) == -1) {
            43         printf("open failed!\n");
            44         return -1;
            45     }
            46 
            47     // set timer to wakeup fcntl
            48     act.sa_handler = hander;
            49     sigemptyset(&act.sa_mask);
            50     act.sa_flags = 0// here, must be zero for wakeup fcntl
            51     sigaction(SIGALRM, &act, &oact);
            52 
            53     int sec = alarm(timeout);
            54 
            55     if (writew_lock(fd) == 0) {
            56         alarm(sec);
            57         // recovery signal handler.
            58         sigaction(SIGALRM, &oact, NULL);
            59        
            60        
            printf("locked OK!\n");
            61        
            62
                     // here, add code about file.
            63 #ifdef _TEST
            64         getchar();
            65         ret = 0;
            66 #else
            67         ret = fn(fname);
            68 #endif
            69 
            70         printf("unlocked!\n");
            71         unlock(fd);
            72     }
            73     else {
            74         alarm(sec);
            75         // recovery signal handler.
            76         sigaction(SIGALRM, &oact, NULL);
            77         // lock failed, because of timeout.
            78         printf("write lock failed\n");
            79         ret = -1;
            80     }
            81 
            82     return ret;
            83 }
            84 
            85 // test code
            86 int func(char *fname)
            87 {
            88     printf("check file:%s \n", fname);
            89     getchar();
            90     return 0;
            91 }
            92 
            93 int main()
            94 {
            95     return lockw("file.lock", func, 5);
            96 }
            97 
            98 


            該程序的原理是,利用了alarm(2)設(shè)置的定時(shí)器,在一定時(shí)間過(guò)后會(huì)產(chǎn)生SIGALRM信號(hào),會(huì)使當(dāng)前正在
            執(zhí)行的系統(tǒng)調(diào)用中斷,導(dǎo)致該系統(tǒng)調(diào)用(fcntl)返回失敗。

            上述代碼有以下的說(shuō)明:
             1. 信號(hào)處理函數(shù)hander是一個(gè)空函數(shù),里面什么也不做。它的存在就是為了接收SIGALRM信號(hào)
             2. sigactionsa_flags成員一定要設(shè)置成0,否則不會(huì)是系統(tǒng)調(diào)用中斷
             3. 為了防止把以前設(shè)置的定時(shí)器破壞,不管是加鎖成功還是失敗都立即恢復(fù)以前的定時(shí)器。
             4. 因?yàn)闉榱私邮?strong>SIGALRM信號(hào),我們?cè)O(shè)置了它的信號(hào)處理函數(shù)。那在加鎖失敗和成功后也要恢復(fù)以前的設(shè)定。

            注:雖然上面的代碼能實(shí)現(xiàn)文件鎖超時(shí)等待的問(wèn)題,但又引入了另一個(gè)問(wèn)題,就是該代碼會(huì)破壞以前設(shè)定的定時(shí)器,即使是后面也恢復(fù)了以前的定時(shí)器設(shè)置,也會(huì)有一些副作用。比如:當(dāng)為了等待其他進(jìn)程釋放文件鎖,傳遞到lockw函數(shù)里的等待時(shí)間(也就是形參timeout)超過(guò)了以前設(shè)定的定時(shí)器觸發(fā)時(shí)間,那這段期間內(nèi)的以前設(shè)定的定時(shí)器就無(wú)效了。也就是說(shuō), 在調(diào)用lockw之前,該進(jìn)程了已經(jīng)設(shè)定了一個(gè)2秒的定時(shí)器, 而這個(gè)進(jìn)程在調(diào)用lockw時(shí)傳遞的timeout時(shí)間為10秒(鎖定的阻塞時(shí)間為10秒), 那么從調(diào)用lockw的那一刻起,2秒的定時(shí)器就無(wú)效了,知道鎖定成功或者失敗為止.

            我的個(gè)人簡(jiǎn)歷第一頁(yè) 我的個(gè)人簡(jiǎn)歷第二頁(yè)
            久久久精品国产免大香伊| 久久ZYZ资源站无码中文动漫 | 亚洲精品国产自在久久| 午夜精品久久久久成人| 午夜天堂av天堂久久久| 国内精品久久久久久99蜜桃| 国产精品永久久久久久久久久| 亚洲精品午夜国产va久久| 97久久综合精品久久久综合| 亚洲国产日韩欧美久久| av无码久久久久久不卡网站| 久久婷婷是五月综合色狠狠| 久久精品国产亚洲网站| 国产欧美久久久精品影院| 久久亚洲综合色一区二区三区| 国内精品伊人久久久久妇| 欧美久久精品一级c片片| 亚洲va中文字幕无码久久不卡| 国产午夜精品久久久久九九| 日韩精品久久久久久免费| 无码8090精品久久一区 | 久久久久九国产精品| 久久精品午夜一区二区福利 | 久久男人中文字幕资源站| 国产精品久久久久9999高清| 亚洲国产精品无码久久| 一日本道伊人久久综合影| 久久综合五月丁香久久激情| 青青青国产成人久久111网站| 99久久精品毛片免费播放| 久久久无码精品亚洲日韩蜜臀浪潮 | 久久久精品国产sm调教网站 | 99久久99久久| 久久精品aⅴ无码中文字字幕不卡 久久精品aⅴ无码中文字字幕重口 | 久久精品免费大片国产大片| 精品多毛少妇人妻AV免费久久| 久久久久久免费一区二区三区| 精品久久久久久国产91| 欧美综合天天夜夜久久| 国产精品99久久久久久董美香| 久久免费线看线看|