1. 程式人生 > >Shell編程-03-Shell中的特殊變量和擴展變量

Shell編程-03-Shell中的特殊變量和擴展變量

wstring lds entos oai 錯誤 通過 read 退格 操作

特殊變量

? ? 在Shell中的特殊變量主要分別兩種位置參數變量狀態變量兩種。

位置參數變量

? ? Shell中的位置參數變量主要是指$0、$1、$#等,主要用於從命令行、函數或腳本執行等地方傳遞參數。詳細說明如下所示:

  • $0:獲取當前執行的Shell腳本的文件名,如腳本中包含路徑,則輸出也包括路徑
  • $n:獲取當前執行的Shell腳本的第n個參數值,如n>9,則需要使用用大括號,如${10},各參數間用空格進行分隔
  • $#:獲取當前執行的Shell腳本傳入參數的總個數
  • $*:獲取當前執行的Shell所有傳入參數,如不加引號則與$@功能相同,如果添加雙引號**"$*",則表示將所有傳入參數視為單個字符串**,相當於"$1 $2 $3"
  • $@:獲取當前執行的Shell所有傳入參數,**如不加引號則與$*功能相同,如果添加雙引號"$@",則表示將所有傳入參數視為獨立的字符串**,相當於"$1" "$2" "$3"

當"$*"和"$@"都添加雙引號時,兩者的功能有所區別;如不加,則功能相同,無區別。

位置參數變量示例

1、示例一:

[root@localhost Test]# cat para.sh
#!/bin/bash
echo $1 $2 $3 $4 $5 $6 $7 $8 $9 ${10} ${11} ${12} ${13} ${14} ${15}
echo ‘$0  is:‘ $0
echo ‘$1  is:‘ $1
echo ‘$12 is:‘ ${12}
echo ‘$#  is:‘ $#
echo ‘$*  is:‘ $*
echo ‘"$*"is:‘ "$*"
echo ‘$@  is:‘ $@
echo ‘"$@"is:‘ "$@"
# 輸出結果
[root@localhost Test]# bash ~/Test/para.sh {a..z}
a b c d e f g h i j k l m n o
$0  is: /root/Test/para.sh
$1  is: a
$12 is: l
$#  is: 26
$*  is: a b c d e f g h i j k l m n o p q r s t u v w x y z
"$*"is: a b c d e f g h i j k l m n o p q r s t u v w x y z
$@  is: a b c d e f g h i j k l m n o p q r s t u v w x y z
"$@"is: a b c d e f g h i j k l m n o p q r s t u v w x y z

1、傳入的參數個數多於腳本定義的參數時,則多出的參數會忽略
2、傳入的參數中如使用雙引號,則會當作一個參數值進行傳遞
3、位置參數如大於9,需要使用${}進行傳遞

2、示例二:

[root@localhost Test]# cat testposition.sh 
#!/bin/bash
echo ‘$# $1 $2 $3 $* $@‘
echo $# $1 $2 $3 $*  $@
echo "************"
echo ‘$*‘
for tmp in $*
 do
   echo $tmp
 done
echo "************"
echo "@@@@@@@@@@@@"
echo ‘$@‘
for temp in $@
 do
  echo $temp
 done
echo "@@@@@@@@@@@@"

echo ‘"*"*"*"*"*"*‘

echo ‘$*‘
for i in "$*"
 do
   echo $i
 done
echo ‘"*"*"*"*"*"*‘

echo ‘"@"@"@"@"@"@‘

echo ‘$@‘
for j in "$@"
 do
  echo $j
 done
echo ‘"@"@"@"@"@"@‘

[root@localhost Test]# bash testposition.sh  "Hello Jack" Welcome "to Shanghai"
$# $1 $2 $3 $* $@
3 Hello Jack Welcome to Shanghai Hello Jack Welcome to Shanghai Hello Jack Welcome to Shanghai
************
$* # 未加雙引號,所以會輸出全部參數,則第一個和第三個參數會拆開
Hello
Jack
Welcome
to
Shanghai
************
@@@@@@@@@@@@
$@ # 未加雙引號,所以會輸出全部參數,則第一個和第三個參數會拆開
Hello
Jack
Welcome
to
Shanghai
@@@@@@@@@@@@
"*"*"*"*"*"*
$* # 添加雙引號後,傳入的參數全部當一個參數進行輸出
Hello Jack Welcome to Shanghai
"*"*"*"*"*"*
"@"@"@"@"@"@
$@ # 添加雙引號後,傳入的參數全部當獨立的參數進行輸出
Hello Jack
Welcome
to Shanghai
"@"@"@"@"@"@

狀態變量

  • $?:獲取上一個命令或腳本的執行狀態值(0:成功,其他:失敗)
  • $$:獲取當前執行的Shell的進程號(PID)
  • $!:獲取上一個在後臺工作的進程的進程號
  • **$_**:獲取在些之前執行的命令或腳本的最後一個參數

以上四個狀態變量,僅$?常用,其他三個了解即可。

在日常使場景中,$?主要用法如下所示:

  • 1、判斷命令和腳本是否執行成功
  • 2、如腳本中調用exit 數字,則會返回該數字給$?
  • 3、如在函數中,則可以通過return 數字將該數字返回給$?

狀態變量示例

1、$?示例:

[root@localhost Test]# ll /etc/profile
-rw-r--r--. 1 root root 1819 4月  11 2018 /etc/profile
[root@localhost Test]# echo $?
0
[root@localhost Test]# ll /etc/profild
ls: 無法訪問/etc/profild: 沒有那個文件或目錄
[root@localhost Test]# echo $?
2

2、$$示例:

[root@localhost Test]# cat testPID.sh
#!/bin/bash
echo $$ > /tmp/test.pid
sleep 300

[root@localhost Test]# bash testPID.sh  & # 將當前腳本調用到後臺執行
[1] 1671
[root@localhost Test]# ps -ef | grep testPID | grep -v grep
root      1671 23706  0 16:37 pts/0    00:00:00 bash testPID.sh # 查詢PID

3、$!示例:
? ? $!功能類似於$$,只不過是獲取上一次執行腳本的PID

[root@localhost Test]# bash testPID.sh  &
[1] 24078
[root@localhost Test]# echo $!
24078 # 打印上一次在後臺執行的進程號
[root@localhost Test]# ps -ef | grep testPID | grep -v grep
root     24078 23706  0 16:42 pts/0    00:00:00 bash testPID.sh

4、$_示例:

[root@localhost Test]# bash para.sh {a..z}
a b c d e f g h i j k l m n o
$0  is: para.sh
$1  is: a
$12 is: l
$#  is: 26
$*  is: a b c d e f g h i j k l m n o p q r s t u v w x y z
"$*"is: a b c d e f g h i j k l m n o p q r s t u v w x y z
$@  is: a b c d e f g h i j k l m n o p q r s t u v w x y z
"$@"is: a b c d e f g h i j k l m n o p q r s t u v w x y z
[root@localhost Test]# echo $_
z # 打印最後一個傳入的參數值

Bash 內置變量

? ? 常用的內部命令有echo、eval、exec、export、read、shift、exit

echo

? ? 主要用於打印信息,其命令格式如下所示:

echo [options] args

常用參數如下所示:

參數選項 說明
-n 不換行輸出內容
-e 解析轉義字符

echo常用轉義字符如下:

轉義字符 說明
\n 換行
\r 回車
\t Tab
\b 退格
\v 縱向制表符
eval

? ? 當Shell程序運行到eval語句時,將讀入參數args,並將它們組合成一個新的命令而後執行。其命令格式如下所示:

eval args
exec

? ? exec主要用於在不創建新的子進程的情況下,轉而執行指定的命令,當指定命令執行完後,則終止該進程。其命令格式如下所示:

exec args

當使用exec打開文件後,read命令每次都會將文件指針移動到下一行進行讀取,直至結束。因此常用來處理文件內容。

read

? ? 從標準輸入讀取變量或字符串等信息並傳遞給其他變量,其命令格式如下所示

read args
shift

? ? 對傳入的位置參數依次向左移動一個位置,並使用位置參數$#減1,直至0為止。其命令格式如下所示:

shift postition args

shift如果不帶參數,則默認左移1位
如傳入的參數為$1 $2 $3,如執行一次shift,則之前的$3變成$2,$2變成$1,$1消失。

exit

? ? 常用於退出Shell,在日常使用過程中可使用exit num來自定義返回狀態數。

Bash 內置變量示例

1、echo

[root@localhost Test]# echo "Test";echo "Dao"
Test
Dao
[root@localhost Test]# echo -n "Test";echo "Dao"
TestDao
[root@localhost Test]# echo -e "Test\tName\n Dao"
Test    Name
 Dao

2、eval

[root@localhost Test]# cat eval.sh
#!/bin/bash

echo "No eval"
echo \$$#
echo "Add eval"
eval echo \$$#

[root@localhost Test]# bash eval.sh  a b
No eval
$2 # 未添加evel時,$#為2,則輸出$2
Add eval
b  # 添加evel後,則重新對傳入的參數進行解析,則輸出傳入的第2個參數

3、exec

[root@localhost Test]# exec ls
eval.sh  para.sh  ping.sh  testPID.sh  testposition.sh
[admin@localhost ~]$  # 在執行exec後則終止當前Shell進程,因此從root用戶退出到普通用戶
# 與read一起讀取文件
[root@localhost ~]# seq 5 > /tmp/rand.log
[root@localhost ~]# cat /tmp/rand.log 
1
2
3
4
5
[root@localhost Test]# cat exec.sh
#!/bin/bash
exec < /tmp/rand.log
while read line
 do
  echo $line
 done
echo "Completed"

[root@localhost Test]# bash exec.sh
1
2
3
4
5
Completed

4、read
可以參考公眾號中read命令一文

5、shift

[root@localhost Test]# cat shift.sh
#!/bin/bash
echo $1 $2 $3 $4 $5
until [ -z $1 ]
do
  echo $@
  shift 1
done
[root@localhost Test]# bash shift.sh {1..5}
1 2 3 4 5
1 2 3 4 5
2 3 4 5
3 4 5
4 5
5

變量擴展

變量擴展說明

? ? Shell中變量擴展說明如下所示:

  • ${var}:返回${var}的內容
  • ${#var}:返回${var}的字符長度
  • ${var:offset}:返回${var}從位置offset之後開始提取字符至結束
  • ${var:offset:length}:返回${var}從offset之後,提取長度為length的字符
  • ${var#word}:返回從${var}開頭開始刪除最短匹配的word子符串
  • ${var##word}:返回從${var}開頭開始刪除最長匹配的word子符串
  • ${var%word}:返回從${var}結尾開始刪除最短匹配的word子符串
  • ${var%%word}:返回從${var}結尾開始刪除最長匹配的word子符串
  • ${var/oldstring/newstring}:使用newstring替換第一個匹配的字符oldstring
  • ${var//oldstring/newstring}:使用newstring替換所有匹配的字符oldstring
  • ${var:-word}:如果變量var的值為空或未賦值,則將word做為返回值,常用於防止變量為空或未定義而導致的異常
  • ${var:=word}:如果變量var的值為空或未賦值,則將word賦值給var並返回其值。
  • ${var:?word}:如果變量var的值為空或未賦值,則將word做為標準錯誤輸出,否則則輸出變量的值,常用於捕捉因變量未定義而導致的錯誤並退出程序
  • ${var:+word}:如果變量var的值為空或未賦值,則什麽都不做,否則word字符將替換變量的值
其中${var:-word}、${var:=word}、${var:?word}、${var:+word}中的冒號也可以省略,則將變量為空或未賦值修改為未賦值,去掉了為空的檢測, 即運算符僅檢測變量是否未賦值

變量擴展示例

[root@localhost init.d]# var="This is test string"
[root@localhost init.d]# echo $var
This is test string
[root@localhost init.d]# echo ${var}
This is test string
[root@localhost init.d]# echo ${#var} # 統計字符長度
19
[root@localhost init.d]# echo ${var:5} # 從第5個位置開始截取字符
is test string
[root@localhost init.d]# echo ${var:5:2} # 從第5個位置開始截取2個字符
is
[root@localhost init.d]# echo ${var#This} # 從開頭刪除最短匹配的字符 is
is test string
[root@localhost init.d]# echo ${var##This}  # 從開頭刪除最長匹配的字符 is
is test string
[root@localhost init.d]# echo ${var%g} # 從結尾刪除最短匹配的字符 is
This is test strin
[root@localhost init.d]# echo ${var%%g} # 從結尾刪除最長匹配的字符 is
This is test strin
[root@localhost init.d]# echo ${var/is/newis} # 替換第一個匹配的字符
Thnewis is test string
[root@localhost init.d]# echo ${var//is/newis} # 替換所有匹配到的字符
Thnewis newis test string

[root@localhost init.d]# echo $centos # 變量未定義

[root@localhost init.d]# echo ${centos:-UNDEFINE} # 變量為空,返回UNDEFINE
UNDEFINE
[root@localhost init.d]# centos="CentOS"
[root@localhost init.d]# echo ${centos:-UNDEFINE} # 變量已經定義,返回變量本身的值
CentOS
[root@localhost init.d]# unset centos # 取消變量值
[root@localhost init.d]# echo $centos

[root@localhost init.d]# result=${centos:=UNDEFINE}
[root@localhost init.d]# echo $result
UNDEFINE
[root@localhost init.d]# echo $centos # 變量值為空,則將UNDEFINE賦值給centos
UNDEFINE
[root@localhost init.d]# unset centos
[root@localhost init.d]# echo ${centos:?can not find variable centos}

-bash: centos: can not find variable centos  # 變量值為空,輸出自定義錯誤信息

[root@localhost init.d]# centos="IS DEFINED"
[root@localhost init.d]# echo ${centos:?can not find variable centos}
IS DEFINED               #變量值已定義,則輸出變量值
[root@localhost init.d]# unset centos
[root@localhost init.d]# echo ${centos:+do nothing} # 變量值為空,什麽都不操作輸出

[root@localhost init.d]# centos="do"
[root@localhost init.d]# echo ${centos:+do nothing}  # 變量已賦值,則輸出自定義的消息
do nothing

本文同步在微信訂閱號上發布,如各位小夥伴們喜歡我的文章,也可以關註我的微信訂閱號:woaitest,或掃描下面的二維碼添加關註:
技術分享圖片

Shell編程-03-Shell中的特殊變量和擴展變量