中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Linux下怎么復用外部shell腳本

發布時間:2022-01-21 13:45:52 來源:億速云 閱讀:109 作者:柒染 欄目:開發技術

這篇文章的內容主要圍繞Linux下怎么復用外部shell腳本進行講述,文章內容清晰易懂,條理清晰,非常適合新手學習,值得大家去閱讀。感興趣的朋友可以跟隨小編一起閱讀吧。希望大家通過這篇文章有所收獲!


在Linux開發中,經常會編寫shell腳本來執行一些任務,通常是一個腳本只做一件事,隨著任務的增加,腳本會越來越多,可復用的地方也會逐漸增加,這時就需要提取出腳本中的公共的功能放到一個通用的腳本中,其他腳本都能復用它

下面介紹shell腳本中如何執行外部腳本,如何調用外部腳本中的函數,以及腳本復用相關的方法

執行外部腳本的方式

假如在當前目錄有 a.sh 腳本,內容如下

#!/bin/bash

echo "a.sh..."

在一個腳本中執行外部腳本主要有以下幾種方式

source 外部腳本名字

在當前目錄下的 b.sh 腳本,內容如下:

#!/bin/bash

source a.sh
echo "b.sh..."

執行 ./b.sh,結果如下

[root@ecs-centos-7 ~]# ./b.sh 
a.sh...
b.sh...

腳本中 source a.sh 命令 會先執行當前目錄下的 a.sh腳本,所以結果會先輸出 a.sh…再輸出 b.sh腳本本身的打印

點號 外部腳本名字

把 b.sh 腳本中執行a.sh腳本的語句修改成 點號 + 空格 + a.sh ,修改之后的腳本內容如下:

注意:點號和a.sh之間一定要加上空格,否則執行的時候會出錯

#!/bin/bash

. a.sh
echo "b.sh..."

執行 ./b.sh,結果如下

[root@ecs-centos-7 ~]# ./b.sh 
a.sh...
b.sh...

在上述腳本中, . a.sh 會先執行a.sh腳本, 結果會先輸出 a.sh…再輸出 b.sh…

sh 外部腳本名字

sh 外部腳本名字 和 ./外部腳本名字 兩種方式是一樣的,選擇哪一種方式都沒問題,下面是以前面一種方式為例說明的

把 b.sh 腳本中 source a.sh修改成 sh a.sh ,修改之后的腳本內容如下:

#!/bin/bash

sh a.sh
echo "b.sh..."

執行 ./b.sh 命令, 結果如下

[root@ecs-centos-7 ~]# ./b.sh 
a.sh...
b.sh...

可以看出,結果輸出和上面兩種方式是一樣的

三種方式的有什么區別
調用外部腳本有 source 外部腳本 、點號 外部腳本、sh 外部腳本 三種方式,它們之間有什么區別呢?

其中,source 外部腳本 和 點號 外部腳本 兩種方式是相同的,當前腳本繼承了外部腳本的全局變量和函數, 相當于把外部腳本的函數和全局變量導入了當前腳本中

修改 a.sh 和 b.sh 腳本, 內容如下

a.sh腳本

#!/bin/bash

VAR_A=10

func_a()
{
  echo "a.sh...pid:$$,param:$1"
}

b.sh腳本

#!/bin/bash

source a.sh 

func_a $1
echo "vara:$VAR_A"
echo "b.sh...pid:$$"

執行 ./b.sh 5 命令,結果如下

[root@ecs-centos-7 ~]# ./b.sh 5
a.sh...pid:21485,param:5
vara:10
b.sh...pid:21485

兩個腳本中的 $$ 是指執行腳本的進程ID,從結果可以看出,a.sh 和 b.sh 都是在同一個進程內執行的,所以在 b.sh 腳本中執行 source a.sh 命令,會把 a.sh 腳本中的全局變量 VAR_A 和函數 func_a導入到 b.sh中

在 b.sh中打印變量 VAR_A,輸出的值和 a.sh中相同,調用 func_a函數,輸出也說明了調用的是 a.sh中的函數

source 外部腳本 、點號 外部腳本 兩種方式是相同的, 所以, 把 b.sh 中 source a.sh 修改成 . a.sh , 執行 ./b.sh 5, 結果依然是相同的

由于 sh 外部腳本的方式是當前腳本和外部腳本在兩個不同的進程中執行,所以當前腳本不能直接使用外部腳本中的函數和全局變量

修改 a.sh 和 b.sh 腳本, 內容如下

a.sh腳本

#!/bin/bash

test_a()
{
  echo "a.sh...test_a"
}

echo "a.sh...pid:$$"

b.sh腳本

#!/bin/bash

sh a.sh

echo "b.sh...pid:$$"

test_a

執行 ./b.sh 命令,結果如下

[root@ecs-centos-7 ~]# ./b.sh 
a.sh...pid:21818
b.sh...pid:21817

./b.sh:行7: test_a: 未找到命令
從結果可以看出,執行 a.sh 和 b.sh 的進程ID是不同的,b.sh腳本進程找不到test_a函數,所以在b.sh中調用test_a 函數會提示 未找到命令

調用外部腳本中的函數

上一節講到 sh 外部腳本 的方式無法直接使用外部腳本中函數和全局變量,下面提供幾種方法可以解決這個問題

case 分支選擇

這種方法類似于程序代碼中的 switch case 語句,通過switch 選擇不同的分支從而執行不同的邏輯,shell腳本中是使用case關鍵字來實現的

a.sh腳本

#!/bin/bash

VAR_A=10

test_a()
{
   echo "test_a..pid:$$,p1:$1,p2:$2"
}
get_var()
{
  echo ${VAR_A}
}

case "$1" in
    ta)
      test_a $2 $3
      ;;
    var)
      get_var
      ;;
   *)
      echo "parameter err..."
esac

b.sh腳本

#!/bin/bash

echo "b.sh...pid:$$"

sh a.sh ta 3 5

ret=$(sh a.sh var)

echo "ret:$ret"

執行 ./b.sh 命令,結果如下

[root@ecs-centos-7 ~]# ./b.sh 
b.sh...pid:24813
test_a..pid:24814,p1:3,p2:5
ret:10

腳本b.sh一開始打印了調用自身的進程ID

sh a.sh ta 3 5 語句是調用a.sh腳本,傳入的三個參數分別是ta, 3, 5 ,執行a.sh時,傳入的第一個參數 ta經過case匹配之后調用 test_a函數,并把剩下的兩個參數 3和5作為參數傳入函數

ret=$(sh a.sh var) 語句時調用a.sh腳本,傳入一個var 參數,經過case匹配之后調用get_var函數,該函數的作用輸出腳本中全局變量VAR_A的值,語句中$()的作用是獲取()中命令的返回值,這里是把a.sh腳本中 get_var函數的返回值賦值給 ret變量,所以該變量的值是 a.sh腳本中全局變量VAR_A的值

說明:如果想要獲取函數的返回值,可以在函數中用 echo 打印相應的輸出值,然后使用$(函數名 參數列表)可以獲取到函數中打印的值,如上面b.sh腳本中 ret=$(sh a.sh var)語句,變量ret的值是 a.sh腳本中 get_var函數輸出的值10

這里需要注意的是, 如果函數中有echo調試日志,那么調試日志也會一起返回

函數調用模板

上面介紹的用 case 關鍵字去匹配調用不同的函數有一個缺點,每次a.sh腳本中增加一個函數的時候,case 就需要添加一個分支,分支里調用不同的函數,還需要注意函數是否有參數傳入以及參數數量是否正確

我們可以在每個供外部調用腳本的尾部加上以下的語句,就可以解決上述問題, 具體語句如下

if [ $# -ge 1 ]; then
   name="$1"
   shift 1
   $name "$@"
fi

上述語句首先判斷調用腳本時傳入的參數數量,只有參數數量大于等于1才有效,傳入的第一個參數表示函數名字,從第二個參數到最后一個參數都會作為參數傳入到函數中

這里的 shift 1 是把傳入腳本的參數左移一個位置,比如:傳入腳本參數有 $1 $2 $3三個參數,左移一個位置之后, $2 移動到 $1 的位置,$3 移動到 $2 的位置,參數數量變為2了

原因: 傳入腳本的參數中,第一個參數是函數名字,從第二個參數起才是函數的參數,如果不做左移處理,第一個參數函數名字也會作為參數傳入到函數中

下面是完整的腳本內容

a.sh腳本

#!/bin/bash

VAR_A=10

test_a()
{
   echo "test_a..pid:$$,p1:$1,p2:$2"
}

get_var()
{
  echo ${VAR_A}
}

if [ $# -ge 1 ]; then
   name="$1"
   shift 1
   $name "$@"
fi

b.sh腳本

#!/bin/bash

echo "b.sh...pid:$$"

sh a.sh test_a 3 5

ret=$(sh a.sh get_var)

執行 ./b.sh 命令,結果如下

[root@ecs-centos-7 ~]# ./b.sh 
b.sh...pid:25086
test_a..pid:25087,p1:3,p2:5
ret:10

可以看出,結果和上面 case 的方法是一樣的

現在其他腳本中都可以通過 sh a.sh 函數名 參數列表 這樣的方式調用 a.sh 腳本中的函數了,通過 $(sh a.sh 函數名 參數列表)的方式獲取 a.sh腳本函數的返回值

兩者的優缺點

與case分支選擇的方式相比,函數調用模板的優點是調用者只需要關心復用的腳本中函數名、函數傳入參數、函數返回值就可以直接使用

缺點是如果有多個腳本都調用了復用腳本中的函數,當復用腳本中函數名變更時,需要修改所有調用了它的地方

函數調用模板方式的缺點恰恰是case分支選擇方式的有點,case分支選擇的方式時根據傳入的字符串參數調用不同的函數,這里的字符串參數相當于函數的別名,只要這個參數保持不變,腳本中的函數名字可以任意變更

上述的優缺點比較只是一個相對的比較,實際應用中下不會很明顯,大部分情況兩種方式都可以使用

感謝你的閱讀,相信你對“Linux下怎么復用外部shell腳本”這一問題有一定的了解,快去動手實踐吧,如果想了解更多相關知識點,可以關注億速云網站!小編會繼續為大家帶來更好的文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

会理县| 衢州市| 盘锦市| 竹溪县| 安仁县| 呈贡县| 邵阳县| 安阳市| 泸溪县| 阿巴嘎旗| 赤峰市| 铜川市| 临洮县| 安福县| 台湾省| 昭觉县| 嫩江县| 白银市| 宜章县| 邮箱| 惠东县| 陵水| 无极县| 阜新市| 建瓯市| 深圳市| 申扎县| 当阳市| 博白县| 上高县| 安丘市| 嘉禾县| 商河县| 和林格尔县| 庆元县| 石家庄市| 环江| 海南省| 密山市| 达日县| 井研县|