多線程中fork的坑

多線程中fork的坑


問題所在

在寫oj的時(shí)候,由于使用了線程池履植,并且在獲取用戶程序運(yùn)行結(jié)果的時(shí)候使用的是管道進(jìn)行子進(jìn)程的標(biāo)準(zhǔn)輸出的獲取阔涉,
最后帶來了一個(gè)問題,就是發(fā)現(xiàn)本來線程池有5個(gè)任務(wù)弃锐,最后調(diào)試信息的打印確沒有5個(gè)袄友,而且個(gè)數(shù)不確定殿托。

初步懷疑

是不是線程池出現(xiàn)了死鎖的情況。

添加線程池任務(wù)

    void addTask(clTask* newTask)
    {
        pthread_mutex_lock(&pthreadMutex);
        allTask.push(newTask);
        pthread_mutex_unlock(&pthreadMutex);
        pthread_cond_broadcast(&pthreadCond);
    }

線程主函數(shù)

void* clPthread::pthreadMain(void *arg)
{
    clPthread& self = *(clPthread*)arg;
    while(1)
    {
        pthread_mutex_lock(&(self.pthreadMutex));
        while(0 == self.allTask.size())
        {
            //阻塞在條件變量上并且解鎖互斥鎖
            pthread_cond_wait(&(self.pthreadCond),&(self.pthreadMutex));
            //條件變量滿足后剧蚣,加鎖
        }
        clTask* dealTask = self.allTask.front();
        self.allTask.pop();
        self.dealTask.push(dealTask);

        pthread_mutex_unlock(&(self.pthreadMutex));
        dealTask->run();
    }
}

這是很標(biāo)準(zhǔn)的的生產(chǎn)者消費(fèi)者模型了支竹,應(yīng)該不會(huì)出現(xiàn)死鎖的情況。但還是懷疑鸠按,所以后來我在每次線程池運(yùn)行的任務(wù)的地方打印調(diào)試信息礼搁,
發(fā)現(xiàn)任務(wù)是都執(zhí)行了的,并且是5次目尖,并沒有出現(xiàn)死鎖的情況馒吴。那么問題可能出現(xiàn)在下面的代碼中了。

找到問題

再往下面走就是編譯和運(yùn)行的函數(shù)了,oj的編譯和運(yùn)行我是采用fork進(jìn)程然后重定向子進(jìn)程的標(biāo)準(zhǔn)輸出來獲取用戶程序的編譯錯(cuò)誤信息和
運(yùn)行情況的饮戳,獲取子進(jìn)程狀態(tài)和回收子進(jìn)程使用的是wait函數(shù)豪治,至于為什么使用wait函數(shù),因?yàn)槲业南敕ㄊ亲尭高M(jìn)程阻塞等待子進(jìn)程的編譯和運(yùn)行情況扯罐。
<b>問題就出現(xiàn)在這里</b>:

由于我是在線程中fork的子進(jìn)程负拟,子進(jìn)程結(jié)束的時(shí)候會(huì)觸發(fā)wait函數(shù)
但是此時(shí)的wait函數(shù)的觸發(fā)其實(shí)不一定是之前的線程fork的子進(jìn)程觸發(fā)的,可能是其他線程fork的子進(jìn)程歹河,因?yàn)樗芯€程fork的子進(jìn)程的ppid都一樣
等于是所有的子進(jìn)程結(jié)束都會(huì)使得wait蘇醒掩浙,但是這個(gè)wait確不是應(yīng)該蘇醒的那一個(gè)。

void clCompiler::compile()
{
    int fd[2];
    int ret = pipe(fd);
    int res;
    pid_t pid;
    chdir(workDir.c_str());
    solution->prepareCode();
    pid = fork();
    if(pid > 0){
        char buf[1024];
        close(fd[1]);
        //問題所在的地方
        wait(&res);
        read(fd[0],buf,sizeof(buf));
        close(fd[0]);
        if(res == 0){
            //編譯成功
            this->status = COMPLIE_OK;

        } else{
            //編譯失敗
            this->status = COMPLIE_ERROR;
        }
        this->complieError = buf;
    }else {
        //查看ppid都是同一個(gè)
        printf("ppid:%d\n",getppid());
        close(fd[0]);
        dup2(fd[1], STDOUT_FILENO);
        dup2(fd[1], STDERR_FILENO);

        this->doCompile();
        close(fd[1]);
        exit(0);
    }
}

如何解決

既然找到問題了秸歧,那么解決也不是很難了厨姚,只需要使用函數(shù)waitpid指定要回收的進(jìn)程id就行。

雖然修改一個(gè)bug很簡(jiǎn)單键菱,但是發(fā)現(xiàn)問題所在才是需要耐心和能力的遣蚀。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市纱耻,隨后出現(xiàn)的幾起案子芭梯,更是在濱河造成了極大的恐慌,老刑警劉巖弄喘,帶你破解...
    沈念sama閱讀 221,820評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件玖喘,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡蘑志,警方通過查閱死者的電腦和手機(jī)累奈,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,648評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來急但,“玉大人澎媒,你說我怎么就攤上這事〔ㄗ” “怎么了戒努?”我有些...
    開封第一講書人閱讀 168,324評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)镐躲。 經(jīng)常有香客問我储玫,道長(zhǎng),這世上最難降的妖魔是什么萤皂? 我笑而不...
    開封第一講書人閱讀 59,714評(píng)論 1 297
  • 正文 為了忘掉前任撒穷,我火速辦了婚禮,結(jié)果婚禮上裆熙,老公的妹妹穿的比我還像新娘端礼。我一直安慰自己禽笑,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,724評(píng)論 6 397
  • 文/花漫 我一把揭開白布蛤奥。 她就那樣靜靜地躺著蒲每,像睡著了一般。 火紅的嫁衣襯著肌膚如雪喻括。 梳的紋絲不亂的頭發(fā)上邀杏,一...
    開封第一講書人閱讀 52,328評(píng)論 1 310
  • 那天,我揣著相機(jī)與錄音唬血,去河邊找鬼望蜡。 笑死,一個(gè)胖子當(dāng)著我的面吹牛拷恨,可吹牛的內(nèi)容都是我干的脖律。 我是一名探鬼主播,決...
    沈念sama閱讀 40,897評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼腕侄,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼小泉!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起冕杠,我...
    開封第一講書人閱讀 39,804評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤微姊,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后分预,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體兢交,經(jīng)...
    沈念sama閱讀 46,345評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,431評(píng)論 3 340
  • 正文 我和宋清朗相戀三年笼痹,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了配喳。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,561評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡凳干,死狀恐怖晴裹,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情救赐,我是刑警寧澤涧团,帶...
    沈念sama閱讀 36,238評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站净响,受9級(jí)特大地震影響少欺,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜馋贤,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,928評(píng)論 3 334
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望畏陕。 院中可真熱鬧配乓,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,417評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至腰埂,卻和暖如春飒焦,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背屿笼。 一陣腳步聲響...
    開封第一講書人閱讀 33,528評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工牺荠, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人驴一。 一個(gè)月前我還...
    沈念sama閱讀 48,983評(píng)論 3 376
  • 正文 我出身青樓休雌,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親肝断。 傳聞我的和親對(duì)象是個(gè)殘疾皇子杈曲,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,573評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容

  • 又來到了一個(gè)老生常談的問題,應(yīng)用層軟件開發(fā)的程序員要不要了解和深入學(xué)習(xí)操作系統(tǒng)呢胸懈? 今天就這個(gè)問題開始担扑,來談?wù)劜?..
    tangsl閱讀 4,134評(píng)論 0 23
  • 1.ios高性能編程 (1).內(nèi)層 最小的內(nèi)層平均值和峰值(2).耗電量 高效的算法和數(shù)據(jù)結(jié)構(gòu)(3).初始化時(shí)...
    歐辰_OSR閱讀 29,417評(píng)論 8 265
  • 且挽蘭芷若,春風(fēng)舞阡陌趣钱。 蝶影繡殘紅魁亦,添香胭脂色。 青絲影翩躚羔挡,婉然躍畫間洁奈。 眾芳猶宣菲,自引蜂蝶伴绞灼。 晚風(fēng)少垂憐...
    古典依舊閱讀 320評(píng)論 0 0
  • 暑假快結(jié)束了利术,最近在做總結(jié),于是低矮,零零散散記下這篇吧印叁,暑假剛開始的四川之旅。內(nèi)江及成都军掂。 沒旅游什么景點(diǎn)轮蜕,主要是在...
    林小刀159閱讀 251評(píng)論 3 3
  • 前些日子讀完了高爾基的人生三部曲――《童年》、《在人間》蝗锥、《我的大學(xué)》跃洛,雖然只是粗粗讀了一遍,卻總有種苦心孤詣編出...
    夢(mèng)見檀木閱讀 688評(píng)論 2 1