在Linux系統的目錄/var/run下面通常咱們都會看到不少的*.pid文件。並且每每新安裝的程序在運行後也會在/var/run目錄下面產生本身的pid文件。那麼這些pid文件有什麼做用呢?它的內容又是什麼呢?編程
(1) pid文件的內容:pid文件爲文本文件,內容只有一行, 記錄了該進程的ID。函數
用cat命令能夠看到。指針
(2) pid文件的做用:防止進程啓動多個副本。只有得到pid文件(固定路徑固定文件名)寫入權限(F_WRLCK)的進程才能正常啓動並把自身的PID寫入該文件中。其它同一個程序的多餘進程則自動退出。繼承
(3) 編程技巧:進程
調用fcntl設置pid文件的鎖定F_SETLK狀態,其中鎖定的標誌位F_WRLCK。ip
若是成功鎖定,則寫入進程當前PID,進程繼續往下執行。ci
若是鎖定不成功,說明已經有一樣的進程在運行了,當前進程結束退出。rem
lock.l_type = F_WRLCK; lock.l_whence = SEEK_SET;cmd
if (fcntl(fd, F_SETLK, &lock) < 0){ //鎖定不成功, 退出...... } sprintf (buf, "%d\n", (int) pid); pidsize = strlen(buf); if ((tmp = write (fd, buf, pidsize)) != (int)pidsize){ //寫入不成功, 退出...... }it
(4) 一些注意事項: i) 若是進程退出,則該進程加的鎖自動失效。 ii) 若是進程關閉了該文件描述符fd, 則加的鎖失效。(整個進程運行期間不能關閉此文件描述符) iii) 鎖的狀態不會被子進程繼承。若是進程關閉則鎖失效而無論子進程是否在運行。 (Locks are associated with processes. A process can only have one kind of lock set for each byte of a given file. When any file descriptor for that file is closed by the process, all of the locks that process holds on that file are released, even if the locks were made using other descriptors that remain open. Likewise, locks are released when a process exits, and are not inherited by child processes created using fork.)
(5) 參考資料: fcntl(文件鎖) 表頭文件 #include <unistd.h> #include <fcntl.h> 函數定義int fcntl(int fd, int cmd, struct flock *lock); 函數解釋fd:文件描寫符 設置的文件描寫符,參數cmd表明欲壟斷的號召 F_DUPFD 複製參數fd的文件描寫符,厲行獲勝則歸來新複製的文件描寫符, F_GETFD 得到close-on-exec符號,若些符號的FD_CLOEXEC位爲0,表明在調用 exec()相干函數時文件將不會關閉 F_SETFD 設置close-on-exec符號,該符號以參數arg的 FD_CLOEXEC位定奪 F_GETFL得到open()設置的符號 F_SETFL改換open()設置的符號 F_GETLK得到文件鎖定的事態,依據lock的描寫,定奪是否上文件鎖 F_SETLK設置文件鎖定的事態,此刻flcok,構造的l_tpye值定然是F_RDLCK、F_WRLCK或F_UNLCK, 萬一沒法發生鎖定,則歸來-1 F_SETLKW 是F_SETLK的阻塞版本,在沒法得到鎖時會進去睡眠事態,萬一可以得到鎖可能捉拿到信號則歸來 參數lock指針爲flock構造指針定義以下 struct flock { ... short l_typejngaoy.com; short l_whence; off_t l_start; 鎖定區域的開關位置 off_t l_len; 鎖定區域的大小 pid_t l_pid; 鎖定動做的歷程 ... }; 1_type有三種事態: F_RDLCK讀取鎖(分享鎖) F_WRLCK寫入鎖(排斥鎖) F_UNLCK解鎖 l_whence也有三種措施 SEEK_SET以文件開始爲鎖定的起始位置 SEEK_CUR以如今文件讀寫位置爲鎖定的起始位置 SEEK_END以文件尾爲鎖定的起始位置 歸來值 獲勝則歸來0,如有訛謬則歸來-1 l_len:加鎖區的長度 l_pid:具備阻塞目前歷程的鎖,其持有歷程的歷程號儲藏在l_pid中,由F_GETLK歸來 等閒是將l_start設置爲0,l_whence設置爲SEEK_SET,l_len設置爲0