0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

關(guān)于 fork 和 exec 是如何在 Unix 上工作的

Linux愛好者 ? 2018-01-22 09:09 ? 次閱讀

本文是關(guān)于 fork 和 exec 是如何在 Unix 上工作的。你或許已經(jīng)知道,也有人還不知道。幾年前當(dāng)我了解到這些時(shí),我驚嘆不已。

我們要做的是啟動(dòng)一個(gè)進(jìn)程。我們已經(jīng)在博客上討論了很多關(guān)于系統(tǒng)調(diào)用的問題,每當(dāng)你啟動(dòng)一個(gè)進(jìn)程或者打開一個(gè)文件,這都是一個(gè)系統(tǒng)調(diào)用。所以你可能會(huì)認(rèn)為有這樣的系統(tǒng)調(diào)用:

start_process(["ls","-l","my_cool_directory"])

這是一個(gè)合理的想法,顯然這是它在 DOS 或 Windows 中的工作原理。我想說的是,這并不是 Linux 上的工作原理。但是,我查閱了文檔,確實(shí)有一個(gè) posix_spawn 的系統(tǒng)調(diào)用基本上是這樣做的,不過這不在本文的討論范圍內(nèi)。

fork 和 exec

Linux 上的 posix_spawn 是通過兩個(gè)系統(tǒng)調(diào)用實(shí)現(xiàn)的,分別是 fork 和 exec(實(shí)際上是 execve),這些都是人們常常使用的。盡管在 OS X 上,人們使用 posix_spawn,而 fork 和 exec 是不提倡的,但我們將討論的是 Linux。

Linux 中的每個(gè)進(jìn)程都存在于“進(jìn)程樹”中。你可以通過運(yùn)行 pstree 命令查看進(jìn)程樹。樹的根是 init,進(jìn)程號(hào)是 1。每個(gè)進(jìn)程(init 除外)都有一個(gè)父進(jìn)程,一個(gè)進(jìn)程都可以有很多子進(jìn)程。

所以,假設(shè)我要啟動(dòng)一個(gè)名為 ls 的進(jìn)程來列出一個(gè)目錄。我是不是只要發(fā)起一個(gè)進(jìn)程 ls 就好了呢?不是的。

我要做的是,創(chuàng)建一個(gè)子進(jìn)程,這個(gè)子進(jìn)程是我(me)本身的一個(gè)克隆,然后這個(gè)子進(jìn)程的“腦子”被吃掉了,變成 ls。

開始是這樣的:

my parent

|- me

然后運(yùn)行 fork(),生成一個(gè)子進(jìn)程,是我(me)自己的一份克?。?/p>

my parent

|- me

|-- cloneof me

然后我讓該子進(jìn)程運(yùn)行 exec("ls"),變成這樣:

my parent

|- me

|-- ls

當(dāng) ls 命令結(jié)束后,我?guī)缀跤肿兓亓宋易约海?/p>

my parent

|- me

|-- ls(zombie)

在這時(shí) ls 其實(shí)是一個(gè)僵尸進(jìn)程。這意味著它已經(jīng)死了,但它還在等我,以防我需要檢查它的返回值(使用 wait 系統(tǒng)調(diào)用)。一旦我獲得了它的返回值,我將再次恢復(fù)獨(dú)自一人的狀態(tài)。

my parent

|- me

fork 和 exec 的代碼實(shí)現(xiàn)

如果你要編寫一個(gè) shell,這是你必須做的一個(gè)練習(xí)。

事實(shí)證明,有了 C 或 Python 的技能,你可以在幾個(gè)小時(shí)內(nèi)編寫一個(gè)非常簡(jiǎn)單的 shell,像 bash 一樣。(至少如果你旁邊能有個(gè)人多少懂一點(diǎn),如果沒有的話用時(shí)會(huì)久一點(diǎn)。)我已經(jīng)完成啦,真的很棒。

這就是 fork 和 exec 在程序中的實(shí)現(xiàn)。我寫了一段 C 的偽代碼。請(qǐng)記住,fork 也可能會(huì)失敗哦。

intpid = fork();

// 我要分身啦

// “我”是誰呢?可能是子進(jìn)程也可能是父進(jìn)程

if(pid == 0){

// 我現(xiàn)在是子進(jìn)程

// “l(fā)s” 吃掉了我腦子,然后變成一個(gè)完全不一樣的進(jìn)程

exec(["ls"])

}elseif(pid == -1){

// 天啊,fork 失敗了,簡(jiǎn)直是災(zāi)難!

}else{

// 我是父進(jìn)程耶

// 繼續(xù)做一個(gè)酷酷的美男子吧

// 需要的話,我可以等待子進(jìn)程結(jié)束

}

上文提到的“腦子被吃掉”是什么意思呢?

進(jìn)程有很多屬性:

打開的文件(包括打開的網(wǎng)絡(luò)連接)

環(huán)境變量

信號(hào)處理程序(在程序上運(yùn)行 Ctrl + C 時(shí)會(huì)發(fā)生什么?)

內(nèi)存(你的“地址空間”)

寄存器

可執(zhí)行文件(/proc/$pid/exe)

cgroups 和命名空間(與 Linux 容器相關(guān))

當(dāng)前的工作目錄

運(yùn)行程序的用戶

其他我還沒想到的

當(dāng)你運(yùn)行execve并讓另一個(gè)程序吃掉你的腦子的時(shí)候,實(shí)際上幾乎所有東西都是相同的! 你們有相同的環(huán)境變量、信號(hào)處理程序和打開的文件等等。

唯一改變的是,內(nèi)存、寄存器以及正在運(yùn)行的程序,這可是件大事。

為何 fork 并非那么耗費(fèi)資源(寫入時(shí)復(fù)制)

你可能會(huì)問:“如果我有一個(gè)使用了 2GB 內(nèi)存的進(jìn)程,這是否意味著每次我啟動(dòng)一個(gè)子進(jìn)程,所有 2 GB 的內(nèi)存都要被復(fù)制一次?這聽起來要耗費(fèi)很多資源!”

事實(shí)上,Linux 為fork()調(diào)用實(shí)現(xiàn)了寫時(shí)復(fù)制copy on write,對(duì)于新進(jìn)程的 2GB 內(nèi)存來說,就像是“看看舊的進(jìn)程就好了,是一樣的!”。然后,當(dāng)如果任一進(jìn)程試圖寫入內(nèi)存,此時(shí)系統(tǒng)才真正地復(fù)制一個(gè)內(nèi)存的副本給該進(jìn)程。如果兩個(gè)進(jìn)程的內(nèi)存是相同的,就不需要復(fù)制了。

為什么你需要知道這么多

你可能會(huì)說,好吧,這些細(xì)節(jié)聽起來很厲害,但為什么這么重要?關(guān)于信號(hào)處理程序或環(huán)境變量的細(xì)節(jié)會(huì)被繼承嗎?這對(duì)我的日常編程有什么實(shí)際影響呢?

有可能哦!比如說,在 Kamal 的博客上有一個(gè)很有意思的bug。它討論了 Python 如何使信號(hào)處理程序忽略了SIGPIPE。也就是說,如果你從 Python 里運(yùn)行一個(gè)程序,默認(rèn)情況下它會(huì)忽略SIGPIPE!這意味著,程序從 Python 腳本和從 shell 啟動(dòng)的表現(xiàn)會(huì)有所不同。在這種情況下,它會(huì)造成一個(gè)奇怪的問題。

所以,你的程序的環(huán)境(環(huán)境變量、信號(hào)處理程序等)可能很重要,都是從父進(jìn)程繼承來的。知道這些,在調(diào)試時(shí)是很有用的。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11304

    瀏覽量

    209503
  • UNIX
    +關(guān)注

    關(guān)注

    0

    文章

    296

    瀏覽量

    41490
  • Fork
    +關(guān)注

    關(guān)注

    0

    文章

    14

    瀏覽量

    3302

原文標(biāo)題:當(dāng)你在 Linux 上啟動(dòng)一個(gè)進(jìn)程時(shí)會(huì)發(fā)生什么?

文章出處:【微信號(hào):LinuxHub,微信公眾號(hào):Linux愛好者】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    AttachedInterrupt如何在NodeMCU上工作?

    我正在嘗試弄清楚 AttachedInterrupt 如何在 NodeMCU 上工作。 我發(fā)現(xiàn)的一切都告訴我這個(gè)代碼沒問題?! void setup() { Serial.begin(9600
    發(fā)表于 07-19 13:39

    Linux下多進(jìn)程編程之fork()函數(shù)語法

    ,因此執(zhí)行速度是比較慢的。為了加快fork()的執(zhí)行速度,很多UNIX系統(tǒng)設(shè)計(jì)者創(chuàng)建了vfork()。vfork()也能創(chuàng)建新進(jìn)程,但它不產(chǎn)生父進(jìn)程的副本。它是通過允許父子進(jìn)程可訪問相同物理內(nèi)存,從而偽裝
    發(fā)表于 08-19 09:28

    【Linux學(xué)習(xí)雜談】之exec族函數(shù)

    ,fork之后調(diào)用exec函數(shù)來執(zhí)行我們的可執(zhí)行程序int execl(const char *path, const char *arg, ...);int execv(const char
    發(fā)表于 09-08 13:14

    使用fork/exec/wait/exit等函數(shù)去創(chuàng)建一個(gè)進(jìn)程

    運(yùn)行。本期課程首先將會(huì)帶領(lǐng)大家了解什么是進(jìn)程,通過編程,學(xué)習(xí)使用fork/exec/wait/exit等函數(shù)去創(chuàng)建一個(gè)進(jìn)程、管理控制一個(gè)進(jìn)程的運(yùn)行、終止一個(gè)進(jìn)程。接下來,會(huì)帶領(lǐng)大家打通進(jìn)程與終端之間的關(guān)系。...
    發(fā)表于 11-04 08:58

    最常見的fork用法是什么

    的進(jìn)程一模一樣,這兩 個(gè)進(jìn)程都會(huì)繼續(xù)運(yùn)行最常見的fork用法是創(chuàng)建一個(gè)新的進(jìn)程,然后使用exec載入二進(jìn)制映像,替換當(dāng)前進(jìn)程的映像。這種情況下,派生(for
    發(fā)表于 12-15 07:38

    esp32在MacBook M1上工作,無法調(diào)試的原因?

    我是 esp32 的新手。在 MacBook M1 上工作。安裝了 ESP-IDF,可以構(gòu)建、刷寫、運(yùn)行和監(jiān)控應(yīng)用程序。從 CLI 和使用 VS 代碼擴(kuò)展。但是我無法調(diào)試。無論我嘗試從 CLI 還是
    發(fā)表于 04-13 06:31

    何在esp8266的LX6上工作?

    個(gè)“NeoPixel ZeroDMA”庫(kù)……但它是特定于 M0/M4 的。 有沒有類似的東西可以在 esp8266 的 LX6 上工作?
    發(fā)表于 05-04 06:58

    何在GitHub上更新Fork以及PullRequest給源項(xiàng)目

    這里有個(gè)小技巧:在默認(rèn)狀態(tài)下,會(huì)是Base源項(xiàng)目,Head我自己的Fork項(xiàng)目;這樣選擇任何一個(gè)時(shí),會(huì)調(diào)到某個(gè)無法更新的頁面;解決方法是先Base或Head一個(gè)其他人的賬號(hào)下的Fork,接著選擇
    的頭像 發(fā)表于 01-08 09:38 ?4374次閱讀
    如<b class='flag-5'>何在</b>GitHub上更新<b class='flag-5'>Fork</b>以及PullRequest給源項(xiàng)目

    對(duì)“Fork”做一個(gè)技術(shù)方面的簡(jiǎn)介

    Linux/Unix 中的進(jìn)程,除了 init 進(jìn)程本身之外,都是由 init 進(jìn)程復(fù)刻fork出來的。關(guān)于服務(wù)器編程方面的復(fù)刻fork的使用,可以進(jìn)一步參閱“搭個(gè) Web 服務(wù)器(
    發(fā)表于 04-02 14:48 ?307次閱讀

    何在Mac終端上使用UNIX命令

    這是在Macintosh計(jì)算機(jī)上運(yùn)行的操作系統(tǒng)。 Mac OS是基于UNIX的Darwin內(nèi)核,因此終端可以讓您基本上直接將命令輸入到UNIX環(huán)境中。
    的頭像 發(fā)表于 08-05 10:00 ?1.1w次閱讀

    信號(hào)通路如何在多層PCB上工作

    在印刷電路板設(shè)計(jì)中,為什么要盡可能使用接地平面?接地平面降低了信號(hào)返回路徑的電感。這反過來又將瞬時(shí)接地電流產(chǎn)生的噪聲降至最低。本文將討論信號(hào)通路如何在多層PCB上工作以及返回通路電感的概念。
    的頭像 發(fā)表于 11-19 17:36 ?2096次閱讀

    通過一個(gè)腳本搞懂fork、source和exec

    Source模式下,子shell執(zhí)行時(shí)獲取的環(huán)境變量會(huì)會(huì)影響到父shell。與fork的區(qū)別在于,不會(huì)額外打開一個(gè)sub-shell來執(zhí)行被調(diào)用的腳本,而是在同一個(gè)shell中執(zhí)行。所以,被調(diào)用的腳本中聲明的變量和環(huán)境變量, 都可以在主腳本中得到和使用。
    的頭像 發(fā)表于 02-03 16:05 ?1738次閱讀

    Qt中的三個(gè)exec之間有什么聯(lián)系

    在Qt中,常見到三個(gè)exec,第一個(gè)是QApplication::exec(),第二個(gè)是QEventLoop::exec,第三個(gè)是QThread::exec()。本文從源碼角度來看看這
    的頭像 發(fā)表于 03-06 09:44 ?2424次閱讀

    Linux中可怕的fork炸彈介紹

    Linux中的Fork炸彈(Fork Bomb)是一種拒絕服務(wù)攻擊的形式,它利用了操作系統(tǒng)中的“fork()”系統(tǒng)調(diào)用。
    的頭像 發(fā)表于 05-22 10:46 ?3024次閱讀
    Linux中可怕的<b class='flag-5'>fork</b>炸彈介紹

    docker exec命令的使用方法

    Docker是一種開源的容器化平臺(tái),可以讓開發(fā)人員在容器中打包和運(yùn)行應(yīng)用程序。它提供了一種快速、可靠和一致的方式來構(gòu)建、部署和運(yùn)行應(yīng)用程序。Docker exec命令是Docker提供的一個(gè)非常
    的頭像 發(fā)表于 11-23 09:33 ?1623次閱讀