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 MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...