• <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++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              18 隨筆 :: 2 文章 :: 10 評論 :: 0 Trackbacks

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

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

             

            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

            之后運行結(jié)果為

            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:從父進程吧數(shù)據(jù)拷貝到子進程中的時候,不按照拷貝構(gòu)造函數(shù)的方式來拷貝,直接進行字節(jié)的拷貝。

            2:對于指針的拷貝,只拷貝指針的值,不拷貝指針?biāo)赶虻膬?nèi)容(malloc 和 new 方式申請的內(nèi)存)。

            另外,當(dāng)進程以 return 的方式結(jié)束的時候,對于沒有釋放對象,系統(tǒng)會調(diào)用將其釋放并調(diào)用他們的析構(gòu)函數(shù),如果以 exit 方式退出則直接釋放內(nèi)存不調(diào)用析構(gòu)函數(shù)。有點類似于 delete 和 free 的區(qū)別。

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

             

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

            評論

            # re: 使用 fork 所要注意的 2008-12-23 17:41 lymons
            >>>>2:對于指針的拷貝,只拷貝指針的值,不拷貝指針?biāo)赶虻膬?nèi)容(malloc 和 new 方式申請的內(nèi)存)。

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

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

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

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

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

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

            久久综合久久鬼色| 久久精品国产亚洲综合色| 久久精品无码一区二区三区| 亚洲AV无码久久精品色欲 | 久久久久久A亚洲欧洲AV冫| 久久免费的精品国产V∧| 亚洲中文字幕无码久久精品1| 久久久午夜精品福利内容| 国产一区二区精品久久凹凸| 精品无码久久久久久久动漫| 91精品国产91热久久久久福利| 久久99精品久久久久久| 久久亚洲欧美日本精品| 91精品国产91久久| 亚洲欧美精品一区久久中文字幕| 久久综合久久鬼色| 亚洲色婷婷综合久久| 精品久久久久久| 久久久久国色AV免费观看| 麻豆精品久久久久久久99蜜桃| 久久久国产精华液| 青青草国产精品久久久久| 精品久久久久久国产免费了| 思思久久好好热精品国产| 久久久久人妻一区二区三区vr| 久久99热狠狠色精品一区| 久久综合伊人77777麻豆| 中文字幕日本人妻久久久免费 | 色婷婷久久综合中文久久一本| 久久久久无码国产精品不卡| 欧美精品国产综合久久| 99久久无色码中文字幕| 久久久久国产一级毛片高清板| 日产精品久久久久久久性色| 国产无套内射久久久国产| 伊人久久大香线蕉AV色婷婷色| 一本大道加勒比久久综合| 久久亚洲日韩看片无码| 国产精品成人久久久久三级午夜电影| 久久一区二区三区免费| 久久久久国产一级毛片高清版|