19.Shell編程進階,數組,字符串,(for,select,while read line)

for循環

for 變量名 in 列表;do
循環體
done前端

  • 執行機制:
    依次將列表中的元素賦值給「變量名」 ; 每次賦值後即執行一次循環體; 直到列表中的元素耗盡,循環結束

列表生成方式:

列表生成方式有多重,詳情可查看其餘博客所總結。算法

  1. 直接給出列表
  2. 整數列表:
    (a) {start..end}
    (b) $(seq [start [step]] end)
  3. 返回列表的命令
    $(COMMAND)
  4. 使用glob通配符,如:*.sh
  5. 變量引用
    $@, $*

for特殊格式

雙小括號方法,即((…))格式,也能夠用於算術運算
雙小括號方法也可使bash Shell實現C語言風格的變量操做
I=10
((I++))shell

  • for循環的特殊格式:
    for ((控制變量初始化;條件判斷表達式;控制變量的修正表達式))
    do
    循環體
    done編程

  • 控制變量初始化:僅在運行到循環代碼段時執行一次
  • 控制變量的修正表達式:每輪循環結束會先進行控制變量修正運算,然後再作條件判斷

這種寫法雙小括號內的判斷是按照C語言風格,而不是-eq 這類shell中test風格的

while循環

while CONDITION; do
循環體
donecentos

  • CONDITION:循環控制條件;進入循環以前,先作一次判斷;每一次循環以後會再次作判斷;條件爲「true」,則執行一次循環;直到條件測試狀態爲「false」終止循環
  • 所以:CONDTION通常應該有循環控制變量;而此變量的值會在循環體不斷地被修正

進入條件:CONDITION爲true
退出條件:CONDITION爲false數組

until循環

until CONDITION; do
循環體
donebash

  • 進入條件: CONDITION 爲false
  • 退出條件: CONDITION 爲true

循環控制語句

continue

用於循環體中
continue [N]:提早結束第N層的本輪循環,而直接進入下一輪判斷;最內層爲第1層服務器

while CONDTIITON1; do
CMD1
...
if CONDITION2; then
continue
fi
CMDn
...
doneapp

break

用於循環體中
break [N]:提早結束第N層循環,最內層爲第1層
while CONDTIITON1; do
CMD1
...
if CONDITION2; then
break
fi
CMDn
...
donessh

注意點:

  1. continue只是結束此這一次循環(默認最內層),而後繼續下一次循環。它仍然在此循環內,只是continue下面的語句在這次循環中再也不執行了。若是後面加上數字,就是跳到數字所在的循環層數內繼續執行數字所在層的後面所剩的循環次數(根據判斷條件判斷的),而比數字小的循環則再也不執行(至關於break掉了比數字小的循環)
  2. break至關於此循環直接中止並跳出(默認最內層),後面加上數字就表明了它直接跳出數字所在層的循環(跳到數字所在層的更外一層,而後執行更外一層的命令,而continue是跳到數字所在層繼續執行數字所在層的剩下的循環)。
  3. 注意break只是循環退出了,並是不exit腳本退出。
  4. 還有一點就是假若有2層或者多層循環,break 退出當前層循環以後它是繼續執行外層循環中在當前循環後面的命令,而contine 2則跳到外層循環以後直接從外層循環的開頭從新進行下一輪循環了。

shift命令

shift [n]
用於將參量列表 list 左移指定次數,缺省爲左移一次。
參量列表 list 一旦被移動,最左端的那個參數就從列表中刪除。 while 循環遍歷位置參量列表時,經常使用到 shift
例如:
./doit.sh a b c d e f g h
./shfit.sh a b c d e f g h

示例:doit.sh
#!/bin/bash
# Name: doit.sh
# Purpose: shift through command line arguments
# Usage: doit.sh [args]
while [ $# -gt 0 ] # or while (( $# > 0 ))
do
echo $*
shift
done

示例:shift.sh
#!/bin/bash
#step through all the positional parameters
until [ -z "$1" ]
do
echo "$1"
shift
done
echo

建立無限循環

while true; do
循環體
done

until false; do
循環體
Done

特殊用法(很實用)

遍歷文件每一行這個寫法很實用,參考本身寫的建立用戶的腳本讀入每一行的命令就知道爲什麼實用了(由於它只把換行符當作分隔符,而不把空格當作分隔符,每一行當作一個長字符串存入變量中)

while循環的特殊用法(遍歷文件的每一行)

while read line; do
循環體
done < /PATH/FROM/SOMEFILE

  • 依次讀取/PATH/FROM/SOMEFILE文件中的每一行,且將行賦值給變量line

select循環與菜單

PS3="Please input a number:"
select variable in list
do
循環體命令
done

  • select 循環主要用於建立菜單,按數字順序排列的菜單項將顯示在標準錯誤上,並顯示 PS3 提示符,等待用戶輸入
  • 用戶輸入菜單列表中的某個數字,執行相應的命令
  • 用戶輸入被保存在內置變量 REPLY 中(和read的同樣)
  • select 是個無限循環,所以要記住用 break 命令退出循環,或用 exit 命令終止腳本。也能夠按 ctrl+c 退出循環
  • select 常常和 case 聯合使用
    注意 :與 for 循環相似,能夠省略 in list,此時使用位置參量
PS3="please input a number:"
select menu in backup clean config start stop status restart quit;do
        case $menu in
        *)
                echo $REPLY
                echo $menu;;
        esac
done

函數介紹

  1. 函數function是由若干條shell命令組成的語句塊,實現代碼重用和模塊化編程
  2. 它與shell程序形式上是類似的,不一樣的是它不是一個單獨的進程,不能獨立運行,而是shell程序的一部分

函數和shell程序比較類似,區別在於

  1. Shell程序在子Shell中運行
  2. 而Shell函數在當前Shell中運行。所以在當前Shell中,函數能夠對shell中變量進行修改

定義函數

函數由兩部分組成:函數名和函數體
help function
語法一:
f_name (){
...函數體...
}
語法二:
function f_name {
...函數體...
}
語法三:
function f_name () {
...函數體...
}

函數使用

  1. 函數的定義和使用:
    可在交互式環境下定義函數
    可將函數放在腳本文件中做爲它的一部分
    可放在只包含函數的單獨文件中
  2. 調用:函數只有被調用纔會執行
    調用:給定函數名
    函數名出現的地方,會被自動替換爲函數代碼
  3. 函數的生命週期:被調用時建立,返回時終止

函數返回值

函數有兩種返回值:

  1. 函數的執行結果返回值:
    (1) 使用echo等命令進行輸出
    (2) 函數體中調用命令的輸出結果
  2. 函數的退出狀態碼:
    (1) 默認取決於函數中執行的最後一條命令的退出狀態碼
    (2) 自定義退出狀態碼,其格式爲:
    return 從函數中返回,用最後狀態命令決定返回值
    return 0 無錯誤返回
    return 1-255 有錯誤返回

環境函數

使子進程也可以使用
聲明:export -f function_name
查看:export -f 或 declare -xf

函數遞歸示例

  • 函數遞歸:
    函數直接或間接調用自身
    注意遞歸層數
  • 示例:階乘

fork×××

fork×××是一種惡意程序,它的內部是一個不斷在fork進程的無限循環,實質是一個簡單的遞歸程序。因爲程序是遞歸的,若是沒有任何限制,這會致使這個簡單的程序迅速耗盡系統裏面的全部資源

  • 函數實現
    :(){ :|:& };:
    bomb() { bomb | bomb & }; bomb
  • 腳本實現
    cat Bomb.sh
    #!/bin/bash
    ./$0|./$0&
func_factorial(){
        local input=$1
        if func_ispositint $input ;then
                [ $input -gt 1 ] && echo $[`func_factorial $[input-1]`*input] || echo 1
        else
                echo -e "Please input a positive integer"
                return 1 
        fi      
}

注意知識點:

  1. 函數就像一個普通命令,聲明定義以後(source以後)直接使用便可,看不出和命令的區別。所以命名函數的時候最好以func_funcname的方式進行命名加以區分。
  2. 刪除函數和刪除變量同樣 用unset func_name命令便可
  3. 注意下面這兩個命令,它倆顯示是已經加載入內存中正在被引用的函數

    • declare -F :只看函數名,後面能夠跟函數名查詢特定的函數,下同
    • declare -f : 看函數的定義內容,也就是/etc/init.d/functions文件中的內容
  4. 能夠把本身經常使用的函數寫在一個文件裏,而後source它引用。
    好比系統中/etc/init.d/function中的actions()函數就可拿來直接用,先source或者.它便可.注意腳本中也要source它,這樣它才能在開啓的子shell中也運行。
action string true :它就會顯示正確結果  action string false : 它就會顯示錯誤結果
  1. 注意由於函數是在當前shell中運行,所以它的變量會影響當前腳本或者shell中的變量結果(也就是兩個變量名字相同會形成變量污染),所以 函數中的變量通常都定義爲 local variable=賦值;

    • 三種常見變量1.普通變量直接賦值無需定義,當前shell中有效;2.export 全局變量(環境變量) 能夠定義的時候賦值,也能夠普通變量賦值完以後再定義,子shell和當前shell有效;3.局部變量local 函數中使用
  2. 三個返回:return 返回出函數;break,continue 返回出循環;exit 返回出shell(腳本) ;

    • 一個等待:wait 後臺命令結束後自動返回前臺並打印PS1提示符
    • 注意全部的返回都是返回到執行這個函數,腳本,循環的上一層環境命令處,返回命令後面的命令(在這個函數,腳本,循環中)則再也不執行了。
  3. { cmd; cdm;} 也能夠看做是一個匿名函數(沒有名字的函數,只能調用一次),更詳細的關於括號的解釋參考其餘博客。
  4. return 後面不加數字的話就是返回上面的最後一個命令的$?值做爲這整個函數的返回值,若是加上數字則把這個數字當作這個函數總體的返回值$?.
  5. 函數相似腳本後面也能夠跟參數,用法如出一轍。注意$*和$@的區別,加上雙引號引用此參數傳遞到子函數或者子腳本中,前者表明一整個字符串做爲一個參數,後者表明分別做爲一個一個參數。不加上雙引號沒區別,這也是由於不加雙引號會把裏面的空格做爲分隔符。

    • 注意若是是腳本中調用函數,則腳本後面的參數調入到函數中時,如若函數後的參數是腳本所用的參數,必定要注意參數用$加數字方式的時候,寫好順序。同時若是想要函數對這些參數進行改變,必定注意函數內部的變量不要定義爲局部變量纔可。
  6. 函數若是寫在腳本中要寫在最前面以供後面的命令調用,注意shell中的函數沒有先聲明後定義的用法,只能寫在最前面,不像C語言能夠先聲明後面再定義。
  7. 注意函數遞歸是先遞歸到最裏層,而後最裏層返回一個返回值,而後在從最裏層往外返回到最外層。所以最裏層必定要寫好返回值的條件,否則會永遠遞歸下去消耗系統資源。遞歸×××就今後而來。
  8. 經過階乘的編寫須要注意遞歸函數的編寫:
    • return返回的值是函數的返回值存在$?中,不能做爲函數名字的結果被上一層遞歸計算
    • 用echo 返回出值,而後就能夠表明這個函數名字當前層級的值,能夠被上一層級來利用它進行計算
    • 注意echo 返回值不能寫在最後,只能寫在遞歸的判斷裏面,不然每次遞歸都會echo 一次,會致使每次遞歸函數名的返回值是兩個(2層) 3個(三層).... 這樣的話就沒辦法把函數自己當作一個數值來進行計算了。
    • 最底層沒有返回結果則會永遠遞歸下去,不斷的開子進程,folk×××就是利用此

信號捕捉trap

  1. trap '觸發指令' 信號
    進程收到系統發出的指定信號後,將執行自定義指令,而不會執行原操做
  2. trap '' 信號
    忽略信號的操做
  3. trap '-' 信號
    恢復原信號的操做
  4. trap -p
    列出自定義信號操做
  5. trap finish EXIT
    當腳本退出時,執行finish函數

注意點:

  1. trap的用處就是在shell(主要體如今腳本)中,讓某些命令沒法進行操做,好比說kill -9 命令等等。能夠保證腳本的各類運行狀態
  2. 還有一種狀況就是腳本執行一半由於某些特殊狀況非正常退出,或者說遇到某些條件判斷退出了,則在退出的時候捕獲到退出信號EXIT(EXIT信號並未在kill -l中列出),則會執行退出時指定的命令或者函數(須要腳本前面提早定義函數等)(If a SIGNAL_SPEC is EXIT (0) ARG is executed on exit from the shell.)
  3. trap的生效在腳本中也是從前到後的,所以trap的信號捕獲命令必定要寫在腳本的最前端。同時後面的信號能夠寫對應的數字標號,特殊的0表示shell(腳本)退出信號。
  4. 注意下面trap中寫的echo命令會打斷sleep,若是想要不打斷,則中間的命令部分什麼也不寫空着最好,這樣int命令將沒辦法執行任何操做了。

trap示例

#!/bin/bash
trap 'echo 「signal:SIGINT"' int
trap -p

for((i=0;i<=10;i++))
do
sleep 1
echo $i
done

trap '' int
trap -p

for((i=11;i<=20;i++))
do
sleep 1
echo $i
done

trap '-' int
trap -p

for((i=21;i<=30;i++))
do
sleep 1
echo $i
done

數組

  1. 變量:存儲單個元素的內存空間
  2. 數組:存儲多個元素的連續的內存空間,至關於多個變量的集合
  3. 數組名和索引
    索引:編號從0開始,屬於數值索引
    • 注意:索引可支持使用自定義的格式,而不只是數值格式,即爲關聯索引,從bash4.0版本以後開始支持

聲明數組:

declare -a ARRAY_NAME 普通數組:能夠不用先聲明,直接使用便可
declare -A ARRAY_NAME 關聯數組:必須先聲明纔可使用,若是先使用了,必須unset以後再從新聲明纔可使用
注意:二者不可相互轉換

數組注意點:

  1. bash的數組支持稀疏格式(索引不連續),很是實用,可是要注意稀疏格式的數組總個數就是有元素的個數,而不是數組的下標的最大值的個數
  2. 注意字符串(也就是變量名)取長度的時候不要加下標${#varialble},只有包含多個元素的數組取長度的時候才加下標${#array[*]}。同時字符串想要取中間的部分字符,只能用${var:#:#}的這種方式,由於var它就至關於數組只有一個元素它自己(或者說這個字符串開頭的地址var[0])
  3. 而數組中的每一個元素中內容想要切片的話則要加上下標${array[INDEX]:#:#},(此時就是把這個帶下標的數組元素名當作一個地址,看做是自己的下標0了)。取數組中的各個元素的話則是${array[*|@]:#:#}的方式。
  4. 不寫下標默認爲0,所以字符串數組(也就是變量)就是表明整個字符串,而數組就是表明數組裏面的第一個元素。
17:47[root@centos7 /etc/sysconfig/network-scripts]# abc=12334
17:54[root@centos7 /etc/sysconfig/network-scripts]# echo ${abc[0]}
12334
17:54[root@centos7 /etc/sysconfig/network-scripts]# echo ${abc}
12334
17:54[root@centos7 /etc/sysconfig/network-scripts]# echo ${#abc}
5

17:57[root@centos7 /data/scriptest]# array=([1]=12wad [5]=ff [6]=cvv [8]=1wd234)
17:57[root@centos7 /data/scriptest]# echo $array

17:57[root@centos7 /data/scriptest]# echo ${array[0]}

17:58[root@centos7 /data/scriptest]# echo ${array[1]}
12wad
17:58[root@centos7 /data/scriptest]# echo ${#array[1]}  :第一個元素的長度
5
17:58[root@centos7 /data/scriptest]# echo ${#array[*]}  :數組總長度,稀疏數組只看有幾個元素就是幾
4
17:58[root@centos7 /data/scriptest]# echo ${array[1]:2:3} :第一個元素內容切片
wad
17:58[root@centos7 /data/scriptest]# echo ${array[*]}
12wad ff cvv 1wd234
20:00[root@centos7 /data/scriptest]# echo ${array[*]:2:2} :取不一樣的數組元素,注意和元素內容切片的區分
ff cvv
20:00[root@centos7 /data/scriptest]# echo ${array[*]:2:3}
ff cvv 1wd234

數組賦值

數組元素的賦值

(1) 一次只賦值一個元素
ARRAY_NAME[INDEX]=VALUE
weekdays[0]="Sunday"
weekdays[4]="Thursday"
(2) 一次賦值所有元素(注意儘可能要加上引號,雖然空格也能夠做爲分隔符)
ARRAY_NAME=("VAL1" "VAL2" "VAL3" ...)
(3) 只賦值特定元素
ARRAY_NAME=([0]="VAL1" [3]="VAL2" ...)
(4) 交互式數組值對賦值(注意有一個-a的選項)
read -a ARRAY

顯示全部數組:declare -a

特殊的賦值方式:

num=({1..10}) :1到10賦值給num[0]到num[9]
file=(*.sh) :當前目錄中sh後綴的文件名賦值給file數組

稀疏方式賦值注意:若是先一個一個賦值,而後又一次賦值特定元素(3中的方式),就算賦值的元素下標並不同,但這樣仍然則會將一個一個賦值的結果給覆蓋掉。可是反過來若是先一次性賦值(3中的方式),而後再一個一個賦值下標不一樣的數組元素,則不會覆蓋掉一次性賦值的元素例子:

19:40[root@centos7 /data]# testarray[2]=12
19:40[root@centos7 /data]# testarray[5]=1234
19:40[root@centos7 /data]# echo ${testarray[2]} 
12
19:40[root@centos7 /data]# declare -a
declare -a testarray='( [2]="12" [5]="1234")'
19:40[root@centos7 /data]# testarray=( [1]=wade [3]=waef )
19:40[root@centos7 /data]# echo ${testarray[2]} 

19:40[root@centos7 /data]# declare -a
declare -a testarray='([1]="wade" [3]="waef")'
上面會覆蓋,而後繼續,下面的不會覆蓋:
19:40[root@centos7 /data]# testarray[2]=12
19:44[root@centos7 /data]# testarray[5]=1234
19:44[root@centos7 /data]# declare -a
declare -a testarray='([1]="wade" [2]="12" [3]="waef" [5]="1234")'

引用數組

  1. 引用數組元素
    ${ARRAY_NAME[INDEX]}
    • 注意:省略[INDEX]表示引用下標爲0的元素
  2. 引用數組全部元素
    ${ARRAY_NAME[*]}
    ${ARRAY_NAME[@]}
  3. 數組的長度(數組中元素的個數)
    ${#ARRAY_NAME[*]}
    ${#ARRAY_NAME[@]}
  4. 刪除數組中的某元素:致使稀疏格式
    unset ARRAY[INDEX]
  5. 刪除整個數組(相似的還有刪除變量,刪除函數,注意都是從內存中刪除掉)
    unset ARRAY

數組數據處理

引用數組中的元素:

  1. 數組切片:
    ${ARRAY[*]:offset:number}
    offset 要跳過的元素個數
    number 要取出的元素個數
  2. 取偏移量以後的全部元素
    ${ARRAY[*]:offset}
  3. 向數組中追加元素(由於下標從0開始):
    ARRAY[${#ARRAY[*]}]=value
  4. 關聯數組:
    declare -A ARRAY_NAME
    ARRAY_NAME=([idx_name1]='val1' [idx_name2]='val2‘...)
    注意:關聯數組必須先聲明再調用

示例

  • 生成10個隨機數保存於數組中,並找出其最大值和最小值
    (優化算法:不要每次都判斷是不是第一個,把第一個賦值寫在for循環外面,而後for循環直接從1開始便可)
    #!/bin/bash
    declare -i min max
    declare -a nums
    for ((i=0;i<10;i++));do
    nums[$i]=$RANDOM
    [ $i -eq 0 ] && min=${nums[$i]} && max=${nums[$i]}&& continue
    [ ${nums[$i]} -gt $max ] && max=${nums[$i]}
    [ ${nums[$i]} -lt $min ] && min=${nums[$i]}
    done
    echo 「All numbers are ${nums[*]}」
    echo Max is $max
    echo Min is $min

示例

  • 編寫腳本,定義一個數組,數組中的元素對應的值是/var/log目錄下全部以.log結尾的文件;統計出其下標爲偶數的文件中的行數之和
#!/bin/bash
#
declare -a files
files=(/var/log/*.log)
declare -i lines=0
for i in $(seq 0 $[${#files[*]}-1]); do
if [ $[$i%2] -eq 0 ];then
let lines+=$(wc -l ${files[$i]} | cut -d' ' -f1)
fi
done
echo "Lines: $lines."

字符串切片

  1. ${#var}:返回字符串變量var的長度
  2. ${var:offset}:返回字符串變量var中從第offset個字符後(不包括第offset個字符)的字符開始,到最後的部分,offset的取值在0 到 ${#var}-1 之間(bash4.2後,容許爲負值)
  3. ${var:offset:number}:返回字符串變量var中從第offset個字符後(不包括第offset個字符)的字符開始,長度爲number的部分
  4. ${var: -length}:取字符串的最右側幾個字符
    注意:冒號後必須有一空白字符
  5. ${var:offset:-length}:從最左側跳過offset字符,一直向右取到距離最右側lengh個字符以前的內容
  6. ${var: -length:-offset}:先從最右側向左取到length個字符開始,再向右取到距離最右側offset個字符之間的內容
    注意:-length前有空格(中間的那個必須有空格,:最後的一項能夠不加空格)

字符串處理

  1. 基於模式取子串
    ${var#*word}:其中word能夠是指定的任意字符
    功能:自左而右,查找var變量所存儲的字符串中,第一次出現的word, 刪除字符串開頭至第一次出現word字符串(含)之間的全部字符
  2. ${var##*word}:同上,貪婪模式,不一樣的是,刪除的是字符串開頭至最後一次由word指定的字符之間的全部內容

    • 示例:
      file=「var/log/messages」
      ${file#/}: log/messages
      ${file##
      /}: messages
  3. ${var%word*}:其中word能夠是指定的任意字符
    功能:自右而左,查找var變量所存儲的字符串中,第一次出現的word, 刪除字符串最後一個字符向左至第一次出現word字符串(含)之間的全部字符

    • 例子:
      file="/var/log/messages"
      ${file%/*}: /var/log
  4. ${var%%word*}:同上,只不過刪除字符串最右側的字符向左至最後一次出現word字符之間的全部字符

查找替換

  1. ${var/pattern/substr}:查找var所表示的字符串中,第一次被pattern所匹配到的字符串,以substr替換之
  2. ${var//pattern/substr}:查找var所表示的字符串中,全部能被pattern所匹配到的字符串,以substr替換之
  3. ${var/#pattern/substr}:查找var所表示的字符串中,行首被pattern所匹配到的字符串,以substr替換之
  4. ${var/%pattern/substr}:查找var所表示的字符串中,行尾被pattern所匹配到的字符串,以substr替換之

查找並刪除

  1. ${var/pattern}:刪除var表示的字符串中第一次被pattern匹配到的字符串,從左往右匹配
  2. ${var//pattern}:刪除var表示的字符串中全部被pattern匹配到的字符串,從左往右匹配
  3. ${var/#pattern}:刪除var表示的字符串中全部以pattern爲行首匹配到的字符串
  4. ${var/%pattern}:刪除var所表示的字符串中全部以pattern爲行尾所匹配到的字符串

字符大小寫轉換

${var^^}:把var中的全部小寫字母轉換爲大寫
${var,,}:把var中的全部大寫字母轉換爲小寫

變量賦值的其餘形式

image

  • 注意裏面的expr是一個字符串,不是一個變量。直接就是雙引號引發來的字符串。

高級變量用法-有類型變量

  • Shell變量通常是無類型的,可是bash Shell提供了declare和typeset兩個命令
  • 它可用於指定變量的類型,兩個命令是等價的

declare [選項] 變量名

-r 聲明或顯示只讀變量
-i 將變量定義爲整型數
-a 將變量定義爲數組
-A 將變量定義爲關聯數組
-f 顯示已定義的全部函數名及其內容
-F 僅顯示已定義的全部函數名
-x 聲明或顯示環境變量和函數 :或者export
-l 聲明變量爲小寫字母 declare –l var=UPPER :即便輸入大寫字母,也會變成小寫字母,下同相反
-u 聲明變量爲大寫字母 declare –u var=lower

eval命令 :很實用,可是要會用它

eval命令將會首先掃描命令行進行全部的置換,而後再執行該命令。該命令適用於那些一次掃描沒法實現其功能的變量.該命令對變量進行兩次掃描

  • 示例:
    [root@server ~]# CMD=whoami
    [root@server ~]# echo $CMD
    whoami
    [root@server ~]# eval $CMD
    root
    [root@server ~]# n=10
    [root@server ~]# echo {0..$n}
    {0..10}
    [root@server ~]# eval echo {0..$n}
    0 1 2 3 4 5 6 7 8 9 10

間接變量引用:能夠增長腳本靈活性,好比變量中存放了一個命令(其實直接寫上$CMDvariable就能直接使用,要結果的話用echo加上反向單引號也能用`$CMDvariable`,)

  • 若是第一個變量的值是第二個變量的名字,從第一個變量引用第二個變量的值就稱爲間接變量引用
  • variable1的值是variable2,而variable2又是變量名,variable2的值爲value,
  • 間接變量引用是指經過variable1得到變量值value的行爲
    variable1=variable2
    variable2=value

  • bash Shell提供了兩種格式實現間接變量引用
    eval tempvar=\$$variable1 :注意不加\的話則會把當前shell的PID顯示出來($$)
    tempvar=${!variable1}
  • 示例:
    [root@server ~]# N=NAME
    [root@server ~]# NAME=wang
    [root@server ~]# N1=${!N}
    [root@server ~]# echo $N1
    wangxiaochun
    [root@server ~]# eval N2=\$$N
    [root@server ~]# echo $N2
    wang

建立臨時文件

mktemp命令:建立並顯示臨時文件,可避免衝突
mktemp [OPTION]... [TEMPLATE]
TEMPLATE: filenameXXX
X至少要出現三個

  • OPTION:
    -d: 建立臨時目錄
    -p DIR或--tmpdir=DIR:指明臨時文件所存放目錄位置
  • 示例:
    mktemp /tmp/testXXX
    tmpdir=`mktemp –d /tmp/testdirXXX`
    mktemp --tmpdir=/testdir testXXXXXX

注意

  1. 它建立臨時文件的時候會有默認的輸出,所以能夠把它命令的結果存入到變量中好比file=`mktemp /data/tempXXXX`
  2. 日常創臨時文件的時候直接就寫上了路徑 ,不過也能夠分開寫 要難過-p 指定dir 而後後面只寫臨時文件名字就好了。不過通常不這麼麻煩
  3. 本身單獨編譯安裝的時候某些文件就要用到臨時文件

安裝複製文件

install命令:

install [OPTION]... [-T] SOURCE DEST 單文件
install [OPTION]... SOURCE... DIRECTORY
install [OPTION]... -t DIRECTORY SOURCE...
install [OPTION]... -d DIRECTORY...建立空目錄

  • 選項:
    -m MODE,默認755
    -o OWNER
    -g GROUP
  • 示例:
    install -m 700 -o wang -g admins srcfile desfile
    install –m 770 –d /testdir/installdir

注意:

  1. install至關於cp ,chown ,chgrp ,chmod的集合
  2. 若是直接install 來進行拷貝,它和cp命令相比,它會讓文件所有加上執行權限(ugo都加上)(並且它默認不是交互式,直接覆蓋)
  3. 命令用法如其名,實現腳本一鍵安裝,修改文件屬性
  4. 注意他能夠直接建立一個新的空文件夾,cp沒有這功能(雖然cp拷貝的時候能夠對應建立文件夾,但不能直接命令來單首創建一個空文件夾)

expect介紹

expect 是由Don Libes基於Tcl( Tool Command Language )語言開發的,主要應用於自動化交互式操做的場景,藉助 expect 處理交互的命令,能夠將交互過程如:ssh登陸,ftp登陸等寫在一個腳本上,使之自動化完成。尤爲適用於須要對多臺服務器執行相同操做的環境中,能夠大大提升系統管理人員的工做效率

它執行的時候會捕獲屏幕上出現的關鍵字,而後根據出現的key來自動提交(輸入)內容

expect命令:須要額外安裝,默認不安裝

  • expect 語法:
    expect [選項] [ -c cmds ] [ [ -[f|b] ] cmdfile ] [ args ]
  • 選項
    -c:從命令行執行expect腳本,默認expect是交互地執行的
    示例:expect -c 'expect "\n" {send "pressed enter\n"}
    -d:能夠輸出輸出調試信息
    示例:expect -d ssh.exp
  • expect中相關命令
    spawn 啓動新的進程
    send 用於向進程發送字符串
    expect 從進程接收字符串
    interact 容許用戶交互 :好比只ssh登錄,而後登陸後繼續手工命令
    exp_continue 匹配多個字符串在執行動做後加此命令

expect最經常使用的語法(tcl語言:模式-動做)

  1. 單一分支模式語法:
    expect 「hi」 {send 「You said hi\n"}
    匹配到hi後,會輸出「you said hi」 ,並換行
  2. 多分支模式語法:
    expect "hi" { send "You said hi\n" } \
    "hehe" { send "Hehe yourself\n" } \
    "bye" { send "Good bye\n" }
    • 匹配hi,hello,bye任意字符串時,執行相應輸出。等同以下:
      expect {
      "hi" { send "You said hi\n"}
      "hehe" { send "Hehe yourself\n"}
      "bye" { send " Good bye\n"}
      }

注意:

  1. expect是一個程序命令包,它裏面包含了一個同名的expect命令。
  2. expect默認交互式,進去以後再輸入命令(好比再輸入spawn expect等)
    示例:
示例
#!/usr/bin/expect
spawn scp /etc/fstab 192.168.8.100:/app
expect {
"yes/no" { send "yes\n";exp_continue }
"password" { send "www.123\n" }
}
expect eof

#!/usr/bin/expect
spawn ssh 192.168.8.100
expect {
"yes/no" { send "yes\n";exp_continue }
"password" { send "www.123\n" }
}
interact
#expect eof

示例:變量,變量設置和賦值用set,後面不加等號
#!/usr/bin/expect
set ip 192.168.8.100
set user root
set password magedu
set timeout 10
spawn ssh $user@$ip
expect {
"yes/no" { send "yes\n";exp_continue }
"password" { send "$password\n" }
}
interact

示例:位置參數:變量從0開始而不是shell中的從1開始
#!/usr/bin/expect
set ip [lindex $argv 0]
set user [lindex $argv 1]
set password [lindex $argv 2]
spawn ssh $user@$ip
expect {
"yes/no" { send "yes\n";exp_continue }
"password" { send "$password\n" }
}
interact
#./ssh3.exp 192.168.8.100 root www

示例:執行多個命令
#!/usr/bin/expect
set ip [lindex $argv 0]
set user [lindex $argv 1]
set password [lindex $argv 2]
set timeout 10
spawn ssh $user@$ip
expect {
"yes/no" { send "yes\n";exp_continue }
"password" { send "$password\n" }
}
expect "]#" { send "useradd haha\n" }
expect "]#" { send "echo www |passwd --stdin haha\n" }
send "exit\n"
expect eof
#./ssh4.exp 192.168.8.100 root www

示例:shell腳本調用expect,就是用多行重定向的方式來使用expect
#!/bin/bash
ip=$1
user=$2
password=$3
expect <<EOF
set timeout 20
spawn ssh $user@$ip
expect {
"yes/no" { send "yes\n";exp_continue }
"password" { send "$password\n" }
}
expect "]#" { send "useradd hehe\n" }
expect "]#" { send "echo www |passwd --stdin hehe\n" }
expect "]#" { send "exit\n" }
expect eof
EOF
#./ssh5.sh 192.168.8.100 root www
相關文章
相關標籤/搜索