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

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

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

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

阿銘linux ? 來源:阿銘linux ? 2023-02-03 16:05 ? 次閱讀

前言

這是一個經(jīng)常在面試時被問到的一個問題,對于剛剛接觸shell的初學(xué)者來說,確實不太好搞明白這三者的區(qū)別,下面我通過兩個腳本來幫助你理解它們。

前置知識點

1)我們所執(zhí)行的任何程序,都是由父進程(parent process)所產(chǎn)生出來的一個子進程(child process),子進程在結(jié)束后,將返回到父進程去。此現(xiàn)像在Linux系統(tǒng)中被稱為fork。當子進程被產(chǎn)生的時候,將會從父進程那里獲得一定的資源分配、以及繼承父進程的環(huán)境( 如環(huán)境變量)。

2)環(huán)境變量大體可以分為三類:

  • 內(nèi)置變量:系統(tǒng)提供,不用定義,不能修改,比如$#,$?,$*,$0等
  • 環(huán)境變量:系統(tǒng)提供,不用定義,可以修改,當前進程及其子進程中使用,比如PATH,PWD,SHELL等
  • 用戶變量(本地變量):用戶定義,可以修改,在當前進程使用,比如var=123等

3)環(huán)境變量只能從父進程到子進程單向繼承。換句話說:在子進程中的環(huán)境如何變更,均不會影響父進程的環(huán)境。

4)先準備兩個示例腳本:

vi 1.sh ##內(nèi)容如下

#!/bin/bash
A=aminglinux
echo "PID for 1.sh before exec/source/fork:$$"
export A
echo "1.sh: $A is $A"


case $1 in
    fork)
        echo "using fork"
        bash 2.sh
        ;;
    source)
        echo "using source"
        source 2.sh
        ;;
    exec)
        echo "using exec"
        exec ./2.sh
        ;;
    *)
        echo "using fork"
        bash 2.sh
        ;;
esac


echo "PID for 1.sh after exec/source/fork:$$"
echo"1.sh:$Ais$A"

vi 2.sh ##內(nèi)容如下

#!/bin/bash
echo "PID for 2.sh: $$"
echo "2.sh get $A=$A from 1.sh"
A=ops
export A
echo "2.sh: $A is $A"

給兩個腳本執(zhí)行權(quán)限

chmod +x 1.sh 2.sh

Fork

Fork,字面上就是派生的意思,在當前shell中(可以是腳本,也可以是命令行終端)去執(zhí)行一個bash命令,那么就會派生一個sub-shell,也就是所謂的子shell。這個過程就是fork。bbd720bc-a398-11ed-bfe3-dac502259ad0.pngFork模式下,子shell會繼承父shell的環(huán)境變量、用戶變量,當子shell結(jié)束時,子shell里面產(chǎn)生的環(huán)境變量并不會帶到父shell中。通過執(zhí)行示例腳本,來驗證上面的結(jié)論:
bash 1.sh fork

bbe4dc84-a398-11ed-bfe3-dac502259ad0.png

1)1.sh的PID為15242也就是父shell的PID,而2.sh的PID為15243,這個是子shell的PID。2)在1.sh里定義了變量A,值為aminglinux,然后fork了一個子shell去執(zhí)行了2.sh,在2.sh里變量A的值是ops,但是當2.sh執(zhí)行完后,再回到1.sh,變量A的值依然是aminglinux。

Source

Source模式下,子shell執(zhí)行時獲取的環(huán)境變量會會影響到父shell。與fork的區(qū)別在于,不會額外打開一個sub-shell來執(zhí)行被調(diào)用的腳本,而是在同一個shell中執(zhí)行。所以,被調(diào)用的腳本中聲明的變量和環(huán)境變量, 都可以在主腳本中得到和使用。

下面來執(zhí)行下示例腳本:

bash 1.sh source

bbff84e4-a398-11ed-bfe3-dac502259ad0.png

1)無論1.sh還是2.sh,PID都是17164,這說明source并不會開啟sub-shell,而是和父shell使用了同一個進程。

2)source 2.sh后,變量A的值變成了ops,而后也被帶到了1.sh里。

Exec

Exec模式下,一旦執(zhí)行了子shell,就不會再去執(zhí)行父shell了。它與fork不同,不需要新開一個sub-shell來執(zhí)行被調(diào)用的腳本,被調(diào)用的腳本與父shell在同一個shell內(nèi)執(zhí)行,這個特性和source一樣。但是使用exec調(diào)用一個新腳本后, 父shell中exec之后的內(nèi)容就不會再執(zhí)行了。

我們來看示例腳本執(zhí)行結(jié)果:

bash 1.sh exec

bc0eb0fe-a398-11ed-bfe3-dac502259ad0.png

1)1.sh和2.sh的PID都是18633,這說明exec和source一樣,并不會開啟sub-shell,而是和父shell使用了同一個進程。

2)exec調(diào)用完2.sh之后,腳本就結(jié)束了,沒有再繼續(xù),這是exec的特性!

總結(jié)

  • source命令: 不創(chuàng)建子進程,在當前Shell進程中執(zhí)行腳本,會將新的環(huán)境變量傳遞到當前shell來。
  • exec命令: 不創(chuàng)建子進程, 在當前Shell進程中執(zhí)行腳本,父腳本中exec行之后的內(nèi)容不會執(zhí)行。
  • fork屬于系統(tǒng)調(diào)用, 會創(chuàng)建一個子進程, 父進程會阻塞等待子進程執(zhí)行結(jié)束, 然后繼續(xù)往下執(zhí)行,子進程里的環(huán)境變量不影響父進程。

審核編輯 :李倩


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

    關(guān)注

    0

    文章

    609

    瀏覽量

    28288
  • Shell
    +關(guān)注

    關(guān)注

    1

    文章

    361

    瀏覽量

    23234
  • 腳本
    +關(guān)注

    關(guān)注

    1

    文章

    384

    瀏覽量

    14793

原文標題:通過一個腳本搞懂fork、source和exec

文章出處:【微信號:aming_linux,微信公眾號:阿銘linux】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

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

    本文是關(guān)于 forkexec 是如何在 Unix 上工作的。你或許已經(jīng)知道,也有人還不知道。幾年前當我了解到這些時,我驚嘆不已。
    的頭像 發(fā)表于 01-22 09:09 ?4945次閱讀

    Linux下多進程編程之exec函數(shù)語法及使用實例

    自己重生?!?如果進程想執(zhí)行另一個程序,那么它就可以調(diào)用fork()函數(shù)新建進程,然后調(diào)
    發(fā)表于 08-14 10:09

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

    )fork創(chuàng)建的子進程是為了單獨地執(zhí)行新程序宏觀上面是父子進程同時進行的。(2)可以利用fork的返回值通過if判斷在當中直接書寫我們的父子進程的內(nèi)容,但是這樣有
    發(fā)表于 09-08 13:14

    使用fork/exec/wait/exit等函數(shù)去創(chuàng)建進程

    運行。本期課程首先將會帶領(lǐng)大家了解什么是進程,通過編程,學(xué)習(xí)使用fork/exec/wait/exit等函數(shù)去創(chuàng)建進程、管理控制
    發(fā)表于 11-04 08:58

    最常見的fork用法是什么

    的進程一模一樣,這兩 進程都會繼續(xù)運行最常見的fork用法是創(chuàng)建新的進程,然后使用exec
    發(fā)表于 12-15 07:38

    Linux系統(tǒng)的fork運行機制分析

    如果其中進程的輸出結(jié)果是“pid1:1001, pid2:1002”,寫出其他進程的輸出結(jié)果(不考慮進程執(zhí)行順序)。 明顯這道題的目的是考察linux下fork的執(zhí)行機制。下面我們通過
    發(fā)表于 04-26 16:26 ?1137次閱讀
    Linux系統(tǒng)的<b class='flag-5'>fork</b>運行機制分析

    對“Fork”做一個技術(shù)方面的簡介

    )。隨著 GitHub 的流行,我們經(jīng)常能在各種文章中見到,“fork 某某項目”等說法,是的,“fork”這個直沒有正式的譯名。其實
    發(fā)表于 04-02 14:48 ?277次閱讀

    最為精簡的Linux Fork炸彈解析

    轉(zhuǎn)自:http://blog.saymagic.cn/2015/03/25/fork-bomb.html Jaromil在2002年設(shè)計了最為精簡的Linux Fork**
    的頭像 發(fā)表于 09-07 16:12 ?1087次閱讀

    SystemVerilog中的fork-join_any

    fork-join_any和fork-join有所不同,fork-join_any的父進程直阻塞,直到任何
    的頭像 發(fā)表于 12-09 09:05 ?2000次閱讀

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

    在Qt中,常見到三exec,第一個是QApplication::exec(),第二是QEventLoop::
    的頭像 發(fā)表于 03-06 09:44 ?2225次閱讀

    分享249拿來即用的shell腳本

    ? 由于腳本案例太多,在此僅展示部分,完整版領(lǐng)取方式請見文末 ? ? ? 249拿來即用的shell腳本! ? 這249Shell腳本
    的頭像 發(fā)表于 07-07 10:52 ?1088次閱讀
    分享249<b class='flag-5'>個</b>拿來即用的shell<b class='flag-5'>腳本</b>

    為什么shell腳本中執(zhí)行命令要加exec?這個exec有什么作用呢?

    正常情況下,如果我們想要在shell腳本中執(zhí)行命令,我們只用將該命令原樣寫到這個shell腳本中就好了。
    的頭像 發(fā)表于 08-21 09:10 ?2704次閱讀
    為什么shell<b class='flag-5'>腳本</b>中執(zhí)行命令要加<b class='flag-5'>exec</b>?這個<b class='flag-5'>exec</b>有什么作用呢?

    編寫點名器腳本

    目的:編寫腳本,可以將準別好的名字合集的.txt傳入腳本,然后每次隨機顯示一個人的姓名
    的頭像 發(fā)表于 11-06 12:41 ?527次閱讀
    編寫<b class='flag-5'>一</b><b class='flag-5'>個</b>點名器<b class='flag-5'>腳本</b>

    linux中source命令的用法

    在Linux中,source命令通常用于重新執(zhí)行剛修改的初始化文件,使之立即生效,而不必注銷并重新登錄。source命令可以用于讀取并執(zhí)行腳本文件中的命令,通常用于在當前shell環(huán)境中重新加載
    的頭像 發(fā)表于 11-08 14:35 ?2050次閱讀

    docker exec命令的使用方法

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