女人自慰AV免费观看内涵网,日韩国产剧情在线观看网址,神马电影网特片网,最新一级电影欧美,在线观看亚洲欧美日韩,黄色视频在线播放免费观看,ABO涨奶期羡澄,第一导航fulione,美女主播操b

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

關于 fork 和 exec 是如何在 Unix 上工作的

Linux愛好者 ? 2018-01-22 09:09 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

本文是關于 fork 和 exec 是如何在 Unix 上工作的。你或許已經知道,也有人還不知道。幾年前當我了解到這些時,我驚嘆不已。

我們要做的是啟動一個進程。我們已經在博客上討論了很多關于系統調用的問題,每當你啟動一個進程或者打開一個文件,這都是一個系統調用。所以你可能會認為有這樣的系統調用:

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

這是一個合理的想法,顯然這是它在 DOS 或 Windows 中的工作原理。我想說的是,這并不是 Linux 上的工作原理。但是,我查閱了文檔,確實有一個 posix_spawn 的系統調用基本上是這樣做的,不過這不在本文的討論范圍內。

fork 和 exec

Linux 上的 posix_spawn 是通過兩個系統調用實現的,分別是 fork 和 exec(實際上是 execve),這些都是人們常常使用的。盡管在 OS X 上,人們使用 posix_spawn,而 fork 和 exec 是不提倡的,但我們將討論的是 Linux。

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

所以,假設我要啟動一個名為 ls 的進程來列出一個目錄。我是不是只要發起一個進程 ls 就好了呢?不是的。

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

開始是這樣的:

my parent

|- me

然后運行 fork(),生成一個子進程,是我(me)自己的一份克隆:

my parent

|- me

|-- cloneof me

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

my parent

|- me

|-- ls

當 ls 命令結束后,我幾乎又變回了我自己:

my parent

|- me

|-- ls(zombie)

在這時 ls 其實是一個僵尸進程。這意味著它已經死了,但它還在等我,以防我需要檢查它的返回值(使用 wait 系統調用)。一旦我獲得了它的返回值,我將再次恢復獨自一人的狀態。

my parent

|- me

fork 和 exec 的代碼實現

如果你要編寫一個 shell,這是你必須做的一個練習。

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

這就是 fork 和 exec 在程序中的實現。我寫了一段 C 的偽代碼。請記住,fork 也可能會失敗哦。

intpid = fork();

// 我要分身啦

// “我”是誰呢?可能是子進程也可能是父進程

if(pid == 0){

// 我現在是子進程

// “ls” 吃掉了我腦子,然后變成一個完全不一樣的進程

exec(["ls"])

}elseif(pid == -1){

// 天啊,fork 失敗了,簡直是災難!

}else{

// 我是父進程耶

// 繼續做一個酷酷的美男子吧

// 需要的話,我可以等待子進程結束

}

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

進程有很多屬性:

打開的文件(包括打開的網絡連接)

環境變量

信號處理程序(在程序上運行 Ctrl + C 時會發生什么?)

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

寄存器

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

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

當前的工作目錄

運行程序的用戶

其他我還沒想到的

當你運行execve并讓另一個程序吃掉你的腦子的時候,實際上幾乎所有東西都是相同的! 你們有相同的環境變量、信號處理程序和打開的文件等等。

唯一改變的是,內存、寄存器以及正在運行的程序,這可是件大事。

為何 fork 并非那么耗費資源(寫入時復制)

你可能會問:“如果我有一個使用了 2GB 內存的進程,這是否意味著每次我啟動一個子進程,所有 2 GB 的內存都要被復制一次?這聽起來要耗費很多資源!”

事實上,Linux 為fork()調用實現了寫時復制copy on write,對于新進程的 2GB 內存來說,就像是“看看舊的進程就好了,是一樣的!”。然后,當如果任一進程試圖寫入內存,此時系統才真正地復制一個內存的副本給該進程。如果兩個進程的內存是相同的,就不需要復制了。

為什么你需要知道這么多

你可能會說,好吧,這些細節聽起來很厲害,但為什么這么重要?關于信號處理程序或環境變量的細節會被繼承嗎?這對我的日常編程有什么實際影響呢?

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

所以,你的程序的環境(環境變量、信號處理程序等)可能很重要,都是從父進程繼承來的。知道這些,在調試時是很有用的。

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • Linux
    +關注

    關注

    87

    文章

    11496

    瀏覽量

    213255
  • UNIX
    +關注

    關注

    0

    文章

    296

    瀏覽量

    42221
  • Fork
    +關注

    關注

    0

    文章

    14

    瀏覽量

    3462

原文標題:當你在 Linux 上啟動一個進程時會發生什么?

文章出處:【微信號:LinuxHub,微信公眾號:Linux愛好者】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    AttachedInterrupt如何在NodeMCU上工作

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

    Linux下多進程編程之fork()函數語法

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

    【Linux學習雜談】之exec族函數

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

    使用fork/exec/wait/exit等函數去創建一個進程

    運行。本期課程首先將會帶領大家了解什么是進程,通過編程,學習使用fork/exec/wait/exit等函數去創建一個進程、管理控制一個進程的運行、終止一個進程。接下來,會帶領大家打通進程與終端之間的關系。...
    發表于 11-04 08:58

    最常見的fork用法是什么

    的進程一模一樣,這兩 個進程都會繼續運行最常見的fork用法是創建一個新的進程,然后使用exec載入二進制映像,替換當前進程的映像。這種情況下,派生(for
    發表于 12-15 07:38

    esp32在MacBook M1上工作,無法調試的原因?

    我是 esp32 的新手。在 MacBook M1 上工作。安裝了 ESP-IDF,可以構建、刷寫、運行和監控應用程序。從 CLI 和使用 VS 代碼擴展。但是我無法調試。無論我嘗試從 CLI 還是
    發表于 04-13 06:31

    何在esp8266的LX6上工作

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

    何在GitHub上更新Fork以及PullRequest給源項目

    這里有個小技巧:在默認狀態下,會是Base源項目,Head我自己的Fork項目;這樣選擇任何一個時,會調到某個無法更新的頁面;解決方法是先Base或Head一個其他人的賬號下的Fork,接著選擇
    的頭像 發表于 01-08 09:38 ?4583次閱讀
    如<b class='flag-5'>何在</b>GitHub上更新<b class='flag-5'>Fork</b>以及PullRequest給源項目

    對“Fork”做一個技術方面的簡介

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

    何在Mac終端上使用UNIX命令

    這是在Macintosh計算機上運行的操作系統。 Mac OS是基于UNIX的Darwin內核,因此終端可以讓您基本上直接將命令輸入到UNIX環境中。
    的頭像 發表于 08-05 10:00 ?1.1w次閱讀

    信號通路如何在多層PCB上工作

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

    通過一個腳本搞懂fork、source和exec

    Source模式下,子shell執行時獲取的環境變量會會影響到父shell。與fork的區別在于,不會額外打開一個sub-shell來執行被調用的腳本,而是在同一個shell中執行。所以,被調用的腳本中聲明的變量和環境變量, 都可以在主腳本中得到和使用。
    的頭像 發表于 02-03 16:05 ?2212次閱讀

    Qt中的三個exec之間有什么聯系

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

    Linux中可怕的fork炸彈介紹

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

    docker exec命令的使用方法

    Docker是一種開源的容器化平臺,可以讓開發人員在容器中打包和運行應用程序。它提供了一種快速、可靠和一致的方式來構建、部署和運行應用程序。Docker exec命令是Docker提供的一個非常
    的頭像 發表于 11-23 09:33 ?2276次閱讀