資源描述:
《實(shí)驗(yàn)三020740309趙玉健》由會(huì)員上傳分享,免費(fèi)在線閱讀,更多相關(guān)內(nèi)容在行業(yè)資料-天天文庫(kù)。
1、實(shí)驗(yàn)三:Windows線程及其調(diào)度觀察一、實(shí)驗(yàn)?zāi)康模鹤屚瑢W(xué)們了解什么是線程,操作系統(tǒng)是如何調(diào)度它們的。二、實(shí)驗(yàn)要求:程序使用C語(yǔ)言實(shí)現(xiàn)。三、實(shí)驗(yàn)過(guò)程及程序?qū)崿F(xiàn)intmain(){ inti; if(fork()==0) { for(i=1;i<3;i++) printf("Thisischildprocess"); } else { for(i=1;i<3;i++) printf("Thisisparentprocess"); }} 執(zhí)行結(jié)果為:ThisischildprocessThisischildproces
2、sThisisparentprocessThisisparentprocess fork在英文中是“分叉”的意思,這個(gè)名字取得很形象。一個(gè)進(jìn)程在運(yùn)行中,如果使用了fork,就產(chǎn)生了另一個(gè)進(jìn)程,于是進(jìn)程就“分叉”了。當(dāng)前進(jìn)程為父進(jìn)程,通過(guò)fork()會(huì)產(chǎn)生一個(gè)子進(jìn)程。對(duì)于父進(jìn)程,fork函數(shù)返回子程序的進(jìn)程號(hào)而對(duì)于子程序,fork函數(shù)則返回零,這就是一個(gè)函數(shù)返回兩次的本質(zhì)。可以說(shuō),fork函數(shù)是Unix系統(tǒng)最杰出的成就之一,它是七十年代Unix早期的開(kāi)發(fā)者經(jīng)過(guò)理論和實(shí)踐上的長(zhǎng)期艱苦探索后取得的成果?! ∪绻覀儼焉鲜龀绦蛑械难h(huán)放的大一
3、點(diǎn):intmain(){ inti; if(fork()==0) { for(i=1;i<10000;i++) printf("Thisischildprocess"); } else { for(i=1;i<10000;i++) printf("Thisisparentprocess"); }} 則可以明顯地看到父進(jìn)程和子進(jìn)程的并發(fā)執(zhí)行,交替地輸出“Thisischildprocess”和“Thisisparentprocess”?! 〈藭r(shí)此刻,我們還沒(méi)有完全理解fork()函數(shù),再來(lái)看下面的一段程序,看看究竟會(huì)
4、產(chǎn)生多少個(gè)進(jìn)程,程序的輸出是什么?intmain(){ inti; for(i=0;i<2;i++) { if(fork()==0) { printf("Thisischildprocess"); } else { printf("Thisisparentprocess"); } }} exec 在Linux中可使用exec函數(shù)族,包含多個(gè)函數(shù)(execl、execlp、execle、execv、execve和execvp),被用于啟動(dòng)一個(gè)指定路徑和文件名的進(jìn)程?! xec函數(shù)族的特點(diǎn)體現(xiàn)在:某進(jìn)程一旦
5、調(diào)用了exec類函數(shù),正在執(zhí)行的程序就被干掉了,系統(tǒng)把代碼段替換成新的程序(由exec類函數(shù)執(zhí)行)的代碼,并且原有的數(shù)據(jù)段和堆棧段也被廢棄,新的數(shù)據(jù)段與堆棧段被分配,但是進(jìn)程號(hào)卻被保留。也就是說(shuō),exec執(zhí)行的結(jié)果為:系統(tǒng)認(rèn)為正在執(zhí)行的還是原先的進(jìn)程,但是進(jìn)程對(duì)應(yīng)的程序被替換了。 fork函數(shù)可以創(chuàng)建一個(gè)子進(jìn)程而當(dāng)前進(jìn)程不死,如果我們?cè)趂ork的子進(jìn)程中調(diào)用exec函數(shù)族就可以實(shí)現(xiàn)既讓父進(jìn)程的代碼執(zhí)行又啟動(dòng)一個(gè)新的指定進(jìn)程,這實(shí)在是很妙的。fork和exec的搭配巧妙地解決了程序啟動(dòng)另一程序的執(zhí)行但自己仍繼續(xù)運(yùn)行的問(wèn)題,請(qǐng)看下面的例子
6、:charcommand[MAX_CMD_LEN];voidmain(){ intrtn;/*子進(jìn)程的返回?cái)?shù)值*/ while(1) { /*從終端讀取要執(zhí)行的命令*/ printf(">"); fgets(command,MAX_CMD_LEN,stdin); command[strlen(command)-1]=0; if(fork()==0) { /*子進(jìn)程執(zhí)行此命令*/ execlp(command,command); /*如果exec函數(shù)返回,表明沒(méi)有正常執(zhí)行命令,打印錯(cuò)誤信息*/ perror(c
7、ommand); exit(errorno); } else { /*父進(jìn)程,等待子進(jìn)程結(jié)束,并打印子進(jìn)程的返回值*/ wait(&rtn); printf("childprocessreturn%d",rtn); } }} 這個(gè)函數(shù)基本上實(shí)現(xiàn)了一個(gè)shell的功能,它讀取用戶輸入的進(jìn)程名和參數(shù),并啟動(dòng)對(duì)應(yīng)的進(jìn)程?! lone clone是Linux2.0以后才具備的新功能,它較fork更強(qiáng)(可認(rèn)為fork是clone要實(shí)現(xiàn)的一部分),可以使得創(chuàng)建的子進(jìn)程共享父進(jìn)程的資源,并且要使用此函數(shù)必須在編譯內(nèi)核
8、時(shí)設(shè)置clone_actually_works_ok選項(xiàng)?! lone函數(shù)的原型為:intclone(int(*fn)(void*),void*child_stack,intflags,void*arg)