色婷婷AⅤ一区二区三区|亚洲精品第一国产综合亚AV|久久精品官方网视频|日本28视频香蕉

          新聞中心

          EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應(yīng)用 > FORK()函數(shù)的理解

          FORK()函數(shù)的理解

          作者: 時間:2012-08-06 來源:網(wǎng)絡(luò) 收藏

          子進程和父進程都執(zhí)行在fork調(diào)用之后的代碼,子進程是父進程的一個拷貝。例如,父進程的數(shù)據(jù)空間、堆棧空間都會給子進程一個拷貝,而不是共享這些內(nèi)存。

          Current implementations don't perform. a complete copy of the parent's data, stack, and heap, since a fork is often followed by an exec. Instead, a technique called copy-on-write (COW) is used. These regions are shared by the parent and the child and have their protection changed by the kernel to read-only. If either process tries to modify these regions, the kernel then makes a copy of that piece of memory only, typically a page in a virtual memory system. Section 9.2 of Bach [1986] and Sections 5.6 and 5.7 of McKusick et al. [1996] provide more detail on this feature.

          我們來給出詳細的注釋

          #include

          #include

          int main(void)

          {

          pid_t pid;

          int count=0;

          /*此處,執(zhí)行fork調(diào)用,創(chuàng)建了一個新的進程, 這個進程共享父進程的數(shù)據(jù)和堆??臻g等,這之后的代碼指令為子進程創(chuàng)建了一個拷貝。 fock 調(diào)用是一個復制進程,fock 不象線程需提供一個做為入口, fock調(diào)用后,新進程的入口就在 fock的下一條語句。*/

          pid = fork();

          /*此處的pid的值,可以說明fork調(diào)用后,目前執(zhí)行的是父進程還是子進程*/

          printf( Now, the pid returned by calling fork() is %dn, pid );

          if ( pid>0 )

          {

          /*當fork在子進程中返回后,fork調(diào)用又向父進程中返回子進程的pid, 如是該段代碼被執(zhí)行,但是注意的事,count仍然為0, 因為父進程中的count始終沒有被重新賦值, 這里就可以看出子進程的數(shù)據(jù)和堆??臻g和父進程是獨立的,而不是共享數(shù)據(jù)*/

          printf( This is the parent process,the child has the pid:%dn, pid );

          printf( In the parent process,count = %dn, count );

          }

          else if ( !pid )

          { /*在子進程中對count進行自加1的操作,但是并沒有影響到父進程中的count值,父進程中的count值仍然為0*/

          printf( This is the child process.n);

          printf( Do your own things here.n );

          count++;

          printf( In the child process, count = %dn, count );

          }

          else

          {

          printf( fork failed.n );

          }

          return 0;

          }

          也就是說,在Linux下一個進程在內(nèi)存里有三部分的數(shù)據(jù),就是代碼段、堆棧段和數(shù)據(jù)段。代碼段,顧名思義,就是存放了程序代碼的數(shù)據(jù),假如機器中有數(shù)個進程運行相同的一個程序,那么它們就可以使用相同的代碼段。堆棧段存放的就是子程序的返回地址、子程序的參數(shù)以及程序的局部變量。而數(shù)據(jù)段則存放程序的全局變量,常數(shù)以及動態(tài)數(shù)據(jù)分配的數(shù)據(jù)空間(比如用malloc之類的取得的空間)。系統(tǒng)如果同時運行數(shù)個相同的程序,它們之間就不能使用同一個堆棧段和數(shù)據(jù)段。

          仔細分析后,我們就可以知道:

          一個程序一旦調(diào)用fork函數(shù),系統(tǒng)就為一個新的進程準備了前述三個段,首先,系統(tǒng)讓新的進程與舊的進程使用同一個代碼段,因為它們的程序還是相同的,對于數(shù)據(jù)段和堆棧段,系統(tǒng)則復制一份給新的進程,這樣,父進程的所有數(shù)據(jù)都可以留給子進程,但是,子進程一旦開始運行,雖然它繼承了父進程的一切數(shù)據(jù),但實際上數(shù)據(jù)卻已經(jīng)分開,相互之間不再有影響了,也就是說,它們之間不再共享任何數(shù)據(jù)了。

          fork()不僅創(chuàng)建出與父進程代碼相同的子進程,而且父進程在fork執(zhí)行點的所有上下文場景也被自動復制到子進程中,包括:

          ——全局和局部變量

          ——打開的文件句柄

          ——共享內(nèi)存、消息等同步對象

          而如果兩個進程要共享什么數(shù)據(jù)的話,就要使用另一套函數(shù)(shmget,shmat,shmdt等)來操作。現(xiàn)在,已經(jīng)是兩個進程了,對于父進程,fork函數(shù)返回了子程序的進程號,而對于子程序,fork函數(shù)則返回零,這樣,對于程序,只要判斷fork函數(shù)的返回值,就知道自己是處于父進程還是子進程中。

          pid控制相關(guān)文章:pid控制原理



          上一頁 1 2 下一頁

          關(guān)鍵詞: 理解 函數(shù) FORK

          評論


          相關(guān)推薦

          技術(shù)專區(qū)

          關(guān)閉