fork函數和vfork函數

来源:https://www.cnblogs.com/mumu597/archive/2020/05/13/12880710.html
-Advertisement-
Play Games

fork函數 在諸多應用中,創建多個進程是任務分解時行之有效的方法。例如,某一網路伺服器進程可在偵聽客戶端請求的同時,為處理每 請求而創建一新的子進程,與此同時,伺服器進程會繼續偵聽更多的客戶端連接請求。以此類手法分解任務,通常會簡化應用程式的設計,同時提高了系統的併發性。(即,可同時處理更多的任務 ...


fork函數

在諸多應用中,創建多個進程是任務分解時行之有效的方法。例如,某一網路伺服器進程可在偵聽客戶端請求的同時,為處理每---請求而創建一新的子進程,與此同時,伺服器進程會繼續偵聽更多的客戶端連接請求。以此類手法分解任務,通常會簡化應用程式的設計,同時提高了系統的併發性。(即,可同時處理更多的任務或請求。)

1 #include <sys/types.h>     
2 #include <unistd.h>    
3 pid_t fork(void); //返回:子進程中為0,父進程中為子進程I D,出錯為-1  

執行調用後將存在兩個進程,且每個進程都會從fork()的返回處繼續執行。

這兩個進程將執行相同的程式文本段,但卻各自擁有不同的棧段、數據段以及堆段拷貝。子進程的棧、數據以及棧段開始時是對父進程記憶體相應各部分的完全複製。執行fork()之後,每個進程均可修改各自的棧數據、以及堆段中的變數,而並不影響另一進程。

註:現在很多的實現並不做一個父進程數據段和堆的完全拷貝,因為在fork之後經常跟隨著exec。作為替代,使用了在寫時複製( Copy-On-Write, COW)的技術。這些區域由父、子進程共用,而且內核將它們的存取許可權改變為只讀的。(詳見Linux|Unix系統編程手冊)

程式代碼則可通過fork()的返回值來區分父、子進程。在父進程中,fork()將 返回新創建子進程的進程ID。

當無法創建子進程時,fork()將返回-1。 失敗的原因可能在於,進程數量要麼超出了系統針對此真實用戶(realuser ID)在進程數量.上所施加的限制(RLIMIT_ NPROC),要麼是觸及允許該系統創建的最大進程數這一系統級上限。

一般來說,在fork之後是父進程先執行還是子進程先執行是不確定的。這取決於內核所使用的調度演算法。如果要求父、子進程之間相互同步,則要求某種形式的進程間通信。

例子:

 1 #include<stdio.h>
 2 #include<sys/types.h>
 3 #include<unistd.h>
 4 #include<sys/wait.h>
 5 #include<stdlib.h>
 6 #include<errno.h>
 7 #include<string.h>
 8 
 9 int main()
10 {
11     pid_t childPid;
12     /* if((childPid=fork())==-1)
13     {
14         printf("Fork error %s\n",strerror(errno));
15         exit(1);
16     }
17     else
18         if(childPid==0)
19         {
20             printf("I am the child : %d\n",getpid());
21             exit(0);
22         }
23         else
24         {
25             printf("I am the father : %d\n",getpid());
26             exit(0);
27         } */
28         switch(childPid=fork()){
29             case -1:
30                 printf("Fork error %s\n",strerror(errno));
31                 exit(1);
32             case 0:
33                 printf("I am the child : %d\n",getpid());
34                 exit(0);
35             default:
36                 printf("I am the father : %d\n",getpid());
37                 exit(0);
38                 
39         }
40     return 0;
41 }
View Code

結果:

 

vfork函數

類似於fork(), vfork()可以為調用進程創建一個新的子進程。然而,vfork()是為子進程立即執行exec()的程式而專門設計的。

#include <sys/types.h>     
#include <unistd.h>    
pid_t vfork(void); //返回:子進程中為0,父進程中為子進程I D,出錯為-1  

vfork()與fork()一樣都創建一個子進程, 但是它並不將父進程的地址空間完全複製到子進程中,因為子進程會立即調用 exec (或exit),於是也就不會存訪該地址空間。不過在子進程調用 exec或exit之前,它在父進程的空間中運行。vfork()和fork()之間的另一個區別是:vfork()保證子進程先運行,在它調用exec或exit之後父進程才可能被調度運行。(如果在調用這兩個函數之前子進程依賴於父進程的進一步動作,則會導致死鎖。)

例子:

 1 #include<stdio.h>
 2 #include<sys/types.h>
 3 #include<unistd.h>
 4 #include<sys/wait.h>
 5 #include<stdlib.h>
 6 #include<errno.h>
 7 #include<string.h>
 8 
 9 int main()
10 {
11     pid_t childPid;
12     if((childPid=vfork())==-1)
13     {
14         printf("Fork error %s\n",strerror(errno));
15         exit(1);
16     }
17     else
18         if(childPid==0)         //子進程
19         {
20             sleep(1);           //子進程睡眠一秒
21             printf("I am the child : %d\n",getpid());
22             exit(0);
23         }
24         else                      //父進程
25         {
26             printf("I am the father : %d\n",getpid());
27             exit(0);
28         }
29     /* switch(childPid=vfork()){
30             case -1:
31                 printf("Fork error %s\n",strerror(errno));
32                 exit(1);
33             case 0:               //子進程
34                 sleep(1);        //子進程睡眠一秒
35                 printf("I am the child : %d\n",getpid());
36                 exit(0);
37             default:            //父進程
38                 printf("I am the father : %d\n",getpid());
39                 exit(0);
40                 
41         } */
42     return 0;
43 }
vfork_pid.c

結果:

 

運行程式時可以看到,程式會停一秒然後分別列印出父子進程的ID.也就是說子進程進來就阻塞一秒,但也沒有先去運行父進程,而是讓子進程運行完了之後才運行父進程。

 

參考資料

Linux/Unix系統編程手冊

Unix環境高級編程

Linux程式設計


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 0. 前言 在上一篇中,我故意留下了查詢的示範沒講。雖然說可以通過以下代碼獲取一個DataReader: 然後通過reader一行一行的讀取數據,但是我並不推薦這樣使用。 在查詢這一高頻需求上,C 為之做了很多工作,提供了更多的選擇。這裡介紹一個查詢的另一套寫法。 1. 離線查詢 C 在查詢上提供了 ...
  • 依賴倒置原則(DIP) 依賴倒置(Dependency Inversion Principle,縮寫DIP)是面向對象六大基本原則之一。他是指一種特定的的解耦形式,使得高層次的模塊不依賴低層次的模塊的實現細節,依賴關係被顛倒(反轉),從而使得低層次模塊依賴於高層次模塊的需求抽象. 該原則規定: 高層 ...
  • 本文屬於OData系列 目錄(可能會有後續修改) "武裝你的WEBAPI OData入門" 武裝你的WEBAPI OData便捷查詢 武裝你的WEBAPI OData分頁查詢 武裝你的WEBAPI OData資源更新 武裝你的WEBAPI OData之EDM 武裝你的WEBAPI OData格式轉換 ...
  • 距離上次提出 [Asp.Net Core] Blazor Server Side 擴展用途 - 配合CEF來製作帶瀏覽器核心的客戶端軟體 的想法後, 差不多2個星期了. 這個玩意也做了一半, 自用是沒問題的, 放出去倒是不夠精細. 如圖: 上面的是開發中的項目文件的截圖. 不是成品. 現在可以用 . ...
  • 這段時間,小豬羅志祥正處於風口浪尖,具體是為啥?還不知道的小伙伴趕緊去補一下最近的娛樂圈八卦~簡單來說,就是我們的小羅同事,以自己超強的體力,以及超強的時間管理能力,重新定義了「多人運動」的含義,重新刷新了大眾對 40 歲男人的印象。 所以啊,像手機這種很有隱私性的產品,一定要設置好各種限制,否則就 ...
  • 1. 破解 2. 安裝支持包 (1)為了訪問matlab時不用加路徑,添加環境變數可在 /etc/profile中添加以下代碼 (2)為了防止可能的錯誤,安裝MATLAB的支持包,在終端輸入以下代碼 3. 建立快捷方式 (1).在網上下載一張matlab的圖片,保存為matlab_logo.png, ...
  • Infi-chu: http://www.cnblogs.com/Infi-chu/ 在top和ps命令中有一列顯示進程狀態,分別有如下值: 值 含義 S 進程處於interruptable sleep D 進程處於uninterruptable sleep R 進程處於運行狀態 Z 進程處於僵屍狀 ...
  • 資源下載: "https://github.com/mengning/mykernel" 實驗內容: 1、配置實驗環境,完成Linux內核編譯。 2、對系統源碼進行修改,基於mykernel 2.0實現一個簡單的操作系統內核。 3、簡要分析操作系統內核核心功能及運行工作機制。 實驗環境: VMWar ...
一周排行
    -Advertisement-
    Play Games
  • 基於.NET Framework 4.8 開發的深度學習模型部署測試平臺,提供了YOLO框架的主流系列模型,包括YOLOv8~v9,以及其系列下的Det、Seg、Pose、Obb、Cls等應用場景,同時支持圖像與視頻檢測。模型部署引擎使用的是OpenVINO™、TensorRT、ONNX runti... ...
  • 十年沉澱,重啟開發之路 十年前,我沉浸在開發的海洋中,每日與代碼為伍,與演算法共舞。那時的我,滿懷激情,對技術的追求近乎狂熱。然而,隨著歲月的流逝,生活的忙碌逐漸占據了我的大部分時間,讓我無暇顧及技術的沉澱與積累。 十年間,我經歷了職業生涯的起伏和變遷。從初出茅廬的菜鳥到逐漸嶄露頭角的開發者,我見證了 ...
  • C# 是一種簡單、現代、面向對象和類型安全的編程語言。.NET 是由 Microsoft 創建的開發平臺,平臺包含了語言規範、工具、運行,支持開發各種應用,如Web、移動、桌面等。.NET框架有多個實現,如.NET Framework、.NET Core(及後續的.NET 5+版本),以及社區版本M... ...
  • 前言 本文介紹瞭如何使用三菱提供的MX Component插件實現對三菱PLC軟元件數據的讀寫,記錄了使用電腦模擬,模擬PLC,直至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1. PLC開發編程環境GX Works2,GX Works2下載鏈接 https:// ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • 1、jQuery介紹 jQuery是什麼 jQuery是一個快速、簡潔的JavaScript框架,是繼Prototype之後又一個優秀的JavaScript代碼庫(或JavaScript框架)。jQuery設計的宗旨是“write Less,Do More”,即倡導寫更少的代碼,做更多的事情。它封裝 ...
  • 前言 之前的文章把js引擎(aardio封裝庫) 微軟開源的js引擎(ChakraCore))寫好了,這篇文章整點js代碼來測一下bug。測試網站:https://fanyi.youdao.com/index.html#/ 逆向思路 逆向思路可以看有道翻譯js逆向(MD5加密,AES加密)附完整源碼 ...
  • 引言 現代的操作系統(Windows,Linux,Mac OS)等都可以同時打開多個軟體(任務),這些軟體在我們的感知上是同時運行的,例如我們可以一邊瀏覽網頁,一邊聽音樂。而CPU執行代碼同一時間只能執行一條,但即使我們的電腦是單核CPU也可以同時運行多個任務,如下圖所示,這是因為我們的 CPU 的 ...
  • 掌握使用Python進行文本英文統計的基本方法,並瞭解如何進一步優化和擴展這些方法,以應對更複雜的文本分析任務。 ...
  • 背景 Redis多數據源常見的場景: 分區數據處理:當數據量增長時,單個Redis實例可能無法處理所有的數據。通過使用多個Redis數據源,可以將數據分區存儲在不同的實例中,使得數據處理更加高效。 多租戶應用程式:對於多租戶應用程式,每個租戶可以擁有自己的Redis數據源,以確保數據隔離和安全性。 ...