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

            MySpace

              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
              18 隨筆 :: 2 文章 :: 10 評論 :: 0 Trackbacks

                在我們需要一個程序成為后臺的守護進程時,一般是通過fork 來創建一個子進程,隨之父進程結束,然后再通過 setsid 來使子進程脫離父進程所屬的進程組和會話。(通俗一點也可以這么認為,我們需要給這個已經長大的孩子重新找個家,并且跟之前家庭斷絕關系,跟人有點類似,雖然關系斷絕了,但親爹是誰這個事實是誰也改變不了的,所以不管發生什么,親爹的ID 號是不會變的。因為之前的父母或家庭都有可能被滅亡導致這個孩子因為沒有家也被餓死。)這樣才能不至于當終端發送 Ctrl+c或是該會話中斷后守護進程也隨之結束,從而達到守護的目的。

                我們一般在寫程序時都有這樣的習慣,先是初始化,然后再開始工作。但是在使用 fork 的時候這一點就要注意了。創建子進程的時候系統會從父進程中將數據拷貝一份到子進程中,如果 fork之前我們 new 或者 malloc 了一塊內存,并且保存了一個指向該內存的指針,那么在創建子進程的時候,也會將這個指針拷貝過去,但此處只拷貝了指針并沒有拷貝內存塊。所以當父進程結束的時候這塊內存會被釋放,那么此時在子進程中的指針則指向了一塊已經被釋放的內存了。有一段這樣的代碼

             

            void test15()

            {

                class A

                {

                public:

                    A():pid_(0)

                    {

                        p_malloc = (char*)malloc(100);

                        printf("pro p_malloc_ = %d \n",p_malloc_);

                    }

                    ~A()

                    {

                        printf("free(p_malloc_) , p_malloc = %d,pid = %d\n", p_malloc_, pid_);

                        free(p_malloc);

                    }

                    A(const A& a)

                    {

                        printf("A(const A& a) \n");

                        p_malloc_ = a.p_malloc_;

                    }

                    void set_pid(int pid)

                    {

                        cout << "set pid" << endl;

                        printf("set pid,pid = %d\n", pid);

                        pid_ = pid;

                    }

                public:

                    char *p_malloc_;

                    int pid_;

             

                };

             

                A a;

             

                pid_t pid;

                pid = fork();

                a.set_pid(pid);

                switch(pid)

                {

                case -1:

                    cout << "fork() is error" << endl;

                    exit(-1);

                    break;

                case 0:

                    //exit(0);

                    break;

                default:

                    setsid();

                    pid_t pre_pid = getppid();

                    printf("child pro pid = %d, pre_pid = %d \n",pid, pre_pid);

             

                    sleep(10);

                    exit(10);

                    

                    break;

             

             

                }

             

            }

            int main(int argc, char *argv[])

            {    test15();

                return 0;

            }

            在 Solrais 下用 CC 編譯

            CC Test.cpp -o Test

            之后運行結果為

            bash-2.03$ Test

            pro p_malloc = 305952

            set pid

            set pid,pid = 23881

            child pro pid = 23881, pre_pid = 22819

            set pid

            set pid,pid = 0

            free(p_malloc) , p_malloc = 305952,pid = 0

             

            我認為說明了幾個問題:

            1:從父進程吧數據拷貝到子進程中的時候,不按照拷貝構造函數的方式來拷貝,直接進行字節的拷貝。

            2:對于指針的拷貝,只拷貝指針的值,不拷貝指針所指向的內容(malloc 和 new 方式申請的內存)。

            另外,當進程以 return 的方式結束的時候,對于沒有釋放對象,系統會調用將其釋放并調用他們的析構函數,如果以 exit 方式退出則直接釋放內存不調用析構函數。有點類似于 delete 和 free 的區別。

            再回到我們上面所談的地方,此時如果有一個連接在一個類的構造函數中實現,斷開連接在它的析構函數中實現。如果這個類在 fork 之前已經 new 了出來,那么當父進程以return的方式結束的時候,它的析構函數就會斷開連接,所以子進程中也無法使用該連接了。這里只是比較隨意的談到了一個例子,可能還會有更多的問題因這個原因而產生,所以在使用 fork 的時候我們要想到這一點從而盡可能的避免 BUG 的產生。

             

            posted on 2008-12-23 14:22 yang-chunlei 閱讀(2436) 評論(4)  編輯 收藏 引用

            評論

            # re: 使用 fork 所要注意的 2008-12-23 17:41 lymons
            >>>>2:對于指針的拷貝,只拷貝指針的值,不拷貝指針所指向的內容(malloc 和 new 方式申請的內存)。

            對于這點我不認同。
            創建出來的子進程是要完全拷貝父進程的內存地址空間(內存印像),也就是子進程的內存和父進程的是一模一樣的,當然也包括在堆(heap)里存放的動態內存。
            所以,除了指針之外,指針指向的內容也是要拷貝出來。
            雖然你打印出來的p_malloc_指向的地址都是相同的,但這個地址是存在于在兩個不同的虛擬地址空間,而且內容是一模一樣的,但它們是兩塊不同的物理內存。所以打印出相同的地址,這并不能說明父子進程的指針是指向同一塊內存的。

            另外,在子進程里對p_malloc_進行任意的讀寫都不會影響到父進程的這塊同樣的內存,反之,父進程來操作地址相同的內存也不會影響到子進程。

            你可以在子進程的代碼里,在sleep(10);之后添加對p_malloc_進行讀寫的語句,看它能否正確執行。請你驗證一下。  回復  更多評論
              

            # re: 使用 fork 所要注意的 2008-12-23 20:29 xiaochong
            fork后的內存肯定要復制的,不過是在寫時復制   回復  更多評論
              

            # re: 使用 fork 所要注意的 2008-12-24 10:26 yang-chunlei
            @lymons
            我測試過了,確實如你所說,當初我沒有想到每個進程都有自己的虛擬空間。但是在我所舉的建立連接的那個例子中,父進程把連接斷開了之后,服務器也隨之斷開了這個連接,雖然子進程中的數據沒有變,但是服務器的鏈接狀態改變了,所以子進程再去進行數據傳輸則會失敗的。還有個問題,如果父進程沒有沒有結束,鏈接也沒有斷開,那么當服務器發送數據過來的時候,該哪個進程去接收呢,因為兩個進程內存中的數據和鏈接狀態都是一樣的,這樣豈不是造成了一個端口被兩個進程監聽的問題。這一點我還沒有去測試,稍后有了答案再進行回復。  回復  更多評論
              

            # re: 使用 fork 所要注意的 2008-12-29 22:44 皇家救星
            對于LS所說的情況,我看有些書是推薦子進程生程后立即關閉從父進程繼承的監聽用的socket  回復  更多評論
              

            青草国产精品久久久久久| 久久久高清免费视频| 久久99精品久久久久久hb无码| 亚洲国产精品无码久久久蜜芽| 欧美黑人激情性久久| 久久九九亚洲精品| 久久综合一区二区无码| 亚洲va久久久噜噜噜久久 | 久久精品国产第一区二区| 人妻丰满?V无码久久不卡| 无码人妻久久一区二区三区免费丨| 久久99精品久久久久婷婷| 欧美久久一级内射wwwwww.| 精品乱码久久久久久久| 亚洲午夜精品久久久久久浪潮| 久久国产免费观看精品3| 性做久久久久久久久老女人| 国内精品久久九九国产精品| 麻豆av久久av盛宴av| 91久久精品电影| 99久久无码一区人妻a黑| 久久亚洲AV无码精品色午夜 | 久久亚洲高清观看| 久久99精品久久久大学生| 久久久久久亚洲精品不卡| 2020久久精品国产免费| 亚洲精品乱码久久久久久蜜桃图片| 久久久久18| 狠狠色综合网站久久久久久久| 久久狠狠高潮亚洲精品| 伊人久久综合无码成人网| 欧美粉嫩小泬久久久久久久 | 开心久久婷婷综合中文字幕| 亚洲国产天堂久久综合网站| 国产人久久人人人人爽| 久久久久久亚洲精品成人| 亚洲综合伊人久久综合| 亚洲成色WWW久久网站| 国内精品综合久久久40p| 久久久久人妻一区二区三区| 久久无码国产专区精品|