使用FD_CLOEXEC實現close-on-exec,關閉子進程無用文件描述符【轉】

轉自:https://blog.csdn.net/chrisniu1984/article/details/7050663

我們經常會碰到需要fork子進程的情況,而且子進程很可能會繼續exec新的程序。這就不得不提到子進程中無用文件描述符的問題!


fork函數的使用本不是這裏討論的話題,但必須提一下的是:子進程以寫時複製(COW,Copy-On-Write)方式獲得父進程的數據空間、堆和棧副本,這其中也包括文件描述符。剛剛fork成功時,父子進程中相同的文件描述符指向系統文件表中的同一項(這也意味着他們共享同一文件偏移量)。


接着,一般我們會調用exec執行另一個程序,此時會用全新的程序替換子進程的正文,數據,堆和棧等。此時保存文件描述符的變量當然也不存在了,我們就無法關閉無用的文件描述符了。所以通常我們會fork子進程後在子進程中直接執行close關掉無用的文件描述符,然後再執行exec。


但是在複雜系統中,有時我們fork子進程時已經不知道打開了多少個文件描述符(包括socket句柄等),這此時進行逐一清理確實有很大難度。我們期望的是能在fork子進程前打開某個文件句柄時就指定好:“這個句柄我在fork子進程後執行exec時就關閉”。其實時有這樣的方法的:即所謂的 close-on-exec。


 close-on-exec的實現只需要調用系統的fcntl就能實現,很簡單幾句代碼就能實現:

  1. int fd=open("foo.txt",O_RDONLY);
  2. int flags = fcntl(fd, F_GETFD);
  3. flags |= FD_CLOEXEC;
  4. fcntl(fd, F_SETFD, flags);
  5.  

這樣,當fork子進程後,仍然可以使用fd。但執行exec後系統就會字段關閉子進程中的fd了。


-------------------------------------------------------- 分割線 ------------------------------------------------------------------------------------

最近好好看了一下open函數,其中flags參數可以傳入O_CLOEXEC標記 [注意:linux 2.6.23纔開始支持此標記]

 

這樣就可以一步實現上面的提到的close-on-exec的效果。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章