• <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 閱讀(2439) 評論(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  回復  更多評論
              

            久久亚洲高清综合| 久久中文字幕无码专区| 中文精品久久久久人妻| 亚洲精品高清国产一线久久| 韩国无遮挡三级久久| 伊人久久综合精品无码AV专区| 欧美亚洲国产精品久久蜜芽| 久久精品国产亚洲av麻豆蜜芽 | 久久精品国产亚洲AV久| 久久免费的精品国产V∧| 人妻无码精品久久亚瑟影视| 久久国产精品免费一区二区三区 | 99精品国产免费久久久久久下载| 久久精品国产亚洲精品2020| 亚洲国产天堂久久综合| 亚洲一本综合久久| 99国内精品久久久久久久| 青青草原综合久久大伊人| 国产精品久久久久一区二区三区| 九九热久久免费视频| 久久香综合精品久久伊人| 国产产无码乱码精品久久鸭| 精品久久久久久亚洲精品| 色悠久久久久久久综合网| 狠狠色综合网站久久久久久久高清| 国产亚洲成人久久| 日韩精品国产自在久久现线拍| 亚洲中文字幕无码久久精品1| 亚洲国产成人久久综合碰| 国产99久久久久久免费看| 国产精品视频久久| 亚洲&#228;v永久无码精品天堂久久| 亚洲AV无码成人网站久久精品大| 狠狠色丁香久久婷婷综合图片| 亚洲日本va午夜中文字幕久久 | 亚洲精品无码久久久久AV麻豆| 爱做久久久久久| 国内精品久久久久久中文字幕| 国产一区二区三精品久久久无广告| 国产福利电影一区二区三区久久老子无码午夜伦不 | 国内精品久久久久影院薰衣草 |