時間:2020-02-12來源:電腦系統城作者:電腦系統城
本篇是快速入門教程,后面的文章再對相關內容進行深入。
Bash只支持單行注釋,使用#
開頭的都被當作注釋語句:
# 整行注釋
echo hello world # 行尾注釋
通過Bash的一些特性,可以取巧實現多行注釋:
: '
注釋1
注釋2
'
: <<'EOF'
注釋1
注釋2
EOF
____='
注釋1
注釋2
'
但是,別閑的蛋疼去用取巧的多行注釋,安心用#
來注釋。
Bash中基本數據類型只有字符串類型,連數值類型都沒有(declare -i
可強制聲明數值類型)。
比如:
# 都會當作字符串
echo haha
echo 1234
Bash中字符串的串聯操作,直接將兩段數據連接在一起即可,不需要任何操作符。
例如:
echo "junma""jinlong"
echo 1234 5678
a=3
echo $a
a="www.junmajinlong.com"
echo $a
a='hello world'
echo $a
Shell中可以引用未定義的變量:
echo $xyzdefabc
可以定義空變量:
a=
echo $a
變量替換是指在命令開始執行前,Shell會先將變量的值替換到引用變量的位置處。
例如:
a="hello"
echo $a world
在echo命令開始執行前,Shell會取得變量a的值hello,并將它替換到命令行的$a
處。于是,在echo命令開始執行時,命令行已經變成:
echo hello world
除了變量替換,Shell還會做其它替換:
這些擴展和替換,都是Shell在調用命令之前就完成的,這和其它語言解析代碼的方式不一樣。
后面會詳細解釋Shell是如何做命令行解析的,如果不掌握命令行解析,當遇到命令行語法錯誤后很可能會花掉大量無謂的時間去調試命令。而掌握命令行解析后,就會對命令生命周期了如執掌,不敢說一次就能寫對所有命令行,但能節省大量調試時間,對寫命令行和寫腳本的能力也會上升一個層次。
使用反引號或$()
可以執行命令替換。
`cmd`
$(cmd)
命令替換是指先執行cmd,將cmd的輸出結果替換到$()
或反引號位置處。
例如:
echo `id root`
echo $(id root)
在echo命令執行前,會先執行id命令,id命令的執行結果:
$ id root
uid=0(root) gid=0(root) groups=0(root)
所以會將結果uid=0(root) gid=0(root) groups=0(root)
替換$(id root)
。于是,echo命令開始執行時,命令行已經變成了:
echo uid=0(root) gid=0(root) groups=0(root)
$[]
或$(())
或let命令可以做算術運算。
let是單獨的命令,不能寫在其它命令行中。
a=3
let a=a+1
echo $a
$[]
和$(())
可以寫在命令行內部,Shell在解析命令行的時候,會對它們做算術運算,然后將運算結果替換到命令行中。
a=33
echo $[a+3]
echo $((a+3))
因為變量替換先于算術替換,所以,使用變量名或引用變量的方式都可以:
a=333
echo $[$a+3]
echo $(($a+3))
每個命令執行后都會有對應的進程退出狀態碼,用來表示該進程是否是正常退出。
所以,在命令行中,在Shell腳本中,經常會使用特殊變量$?
判斷最近一個前臺命令是否正常退出。
通常情況下,如果$?
的值:
另外,在Shell腳本中,所有條件判斷(比如if語句、while語句)都以0退出狀態碼表示True,以非0退出狀態碼為False。
exit命令可用于退出當前Shell進程,比如退出當前Shell終端、退出Shell腳本,等等。
exit [N]
exit可指定退出狀態碼N,如果省略N,則默認退出狀態碼為0,即表示正確退出。
在命令的結尾使用&
符號,可以將這個命令放入后臺執行。
命令放入后臺后,會立即回到Shell進程,Shell進程會立即執行下一條命令(如果有)或退出。
使用$!
可以獲取最近一個后臺進程的PID。
sleep 20 &
echo $!
使用wait
命令可以等待后臺進程(當前Shell進程的子進程)完成:
wait [n1 n2 n3 ...]
不給定任何參數時,會等待所有子進程(即所有后臺進程)完成。
sleep 5 &
wait
echo haha
Shell中有多種組合多個命令的方式。
1.cmd1退出后,執行cmd2
cmd1;cmd2
2.cmd1正確退出(退出狀態碼為0)后,執行cmd2
cmd1 && cmd2
3.cmd1不正確退出后,執行cmd2
cmd1 || cmd2
4.邏輯結合:&&
和||
可以隨意結合
# cmd1正確退出后執行cmd2,cmd2正確退出后執行cmd3
cmd1 && cmd2 && cmd3...
# cmd1正確退出則執行cmd2,cmd1不正確退出會執行cmd3
# cmd1正確退出,但cmd2不正確退出,也會執行cmd3
cmd1 && cmd2 || cmd3
# cmd1正確退出會執行cmd3
# cmd1不正確退出會執行cmd2,cmd2正確退出會執行cmd3
cmd1 || cmd2 && cmd3
5.將多個命令分組:小括號或大括號可以組合多個命令
# 小括號組合的多個命令是在子Shell中執行
# 即會先創建一個新的Shell進程,在執行里面的命令
(cmd1;cmd2;cmd3)
# 大括號組合的多個命令是在當前Shell中執行
# 大括號語法特殊,要求:
# 1.開閉括號旁邊都有空白,否則語法解析錯誤(解析成大括號擴展)
# 2.寫在同一行時,每個cmd后都要加分號結尾
# 3.多個命令可分行書寫,不要求分號結尾
{ cmd1;cmd2;cmd3; }
{
cmd1
cmd2
cmd3
}
軟件設計認為,程序應該有一個數據來源、數據出口和報告錯誤的地方。在Linux系統中,每個程序默認都會打開三個文件描述符(file descriptor,fd):
文件描述符,說白了就是系統為了跟蹤打開的文件而分配給它的一個數字,這個數字和文件有對應關系:從文件描述符讀取數據,即表示從對應的文件中讀取數據,向文件描述符寫數據,即表示向對應文件中寫入數據。
Linux中萬物皆文件,文件描述符也是文件。默認:
這些文件默認又是各個終端的軟鏈接文件:
$ ls -l /dev/std*
lrwxrwxrwx 1 root root 15 Jan 8 20:26 /dev/stderr -> /proc/self/fd/2
lrwxrwxrwx 1 root root 15 Jan 8 20:26 /dev/stdin -> /proc/self/fd/0
lrwxrwxrwx 1 root root 15 Jan 8 20:26 /dev/stdout -> /proc/self/fd/1
$ ls -l /proc/self/fd/
lrwx------ 1 root root 64 Jan 16 10:40 0 -> /dev/pts/0
lrwx------ 1 root root 64 Jan 16 10:40 1 -> /dev/pts/0
lrwx------ 1 root root 64 Jan 16 10:40 2 -> /dev/pts/0
lr-x------ 1 root root 64 Jan 16 10:40 3 -> /proc/75220/fd
所以,默認情況下讀寫數據都是終端,例如:
# 數據輸出到終端
$ echo haha
$ cat /etc/fstab
# 從終端讀取數據
$ cat
hello # 在終端輸入
hello # 在終端輸出
world # 在終端輸入
world # 在終端輸出
^C
改變文件描述符對應的目標,可以改變數據的流向。比如標準輸入fd=1默認流向是終端設備,若將其改為/tmp/a.log,便能讓數據寫入/tmp/a.log文件中而不再是終端設備中。
在Shell中,這種改變文件描述符目標的行為稱為重定向,即重新確定數據的流向。
其實,文件描述符有很多類操作,包括fd的重定向、fd的分配(open,即打開文件)、fd復制(duplicate)、fd的移動(move)、fd的關閉(close)?,F在只介紹基礎重定向操作。
Shell中,基礎重定向操作有以下幾種方式:
另外,經常用于輸出的一個特殊目標文件是/dev/null,它是空設備,可以直接丟掉所有寫入它的數據。
echo www.junmajinlong.com >/dev/null
curl -I www.junmajinlong.com 2>/dev/null >/tmp/a.log
cat </etc/fstab
一個經常用的技巧是清空文件的方式:
$ cat /dev/null >file
$ >file
cat是一個命令,這個命令的源代碼中寫了一些代碼用來處理選項和參數。
cat -n /etc/fstab
cat命令開始執行后,會識別-n
選項,該選項會讓cat輸出時同時輸出行號,cat同時還會識別/etc/fstab
參數,cat會讀取參數指定的文件然后輸出。
如果沒有指定cat的文件參數,則cat默認會從標準輸入中讀取數據。默認的標準輸入是終端,所以在沒有改變標準輸入的流向時,會從終端讀取數據,也就是用戶輸入什么字符,就讀取什么字符,然后輸出什么字符:
$ cat
junmajinlong # 在終端輸入
junmajinlong # 在終端輸出
junma # 在終端輸入
junma # 在終端輸出
^C
但用戶可以改變標準輸入的來源。比如:
$ cat </etc/fstab
表示將標準輸入來源改為/etc/fstab文件,于是cat會從/etc/fstab中讀取數據。
另外,約定俗成的,會使用一個-
來表示標準輸入或標準輸出。
# 下面是等價的,都表示從標準輸入中讀取數據
cat -
cat /dev/stdin
cat
注:這并非是一貫正確的,只是約定俗成的大多數程序的代碼中都定義了-
相關的代碼處理??蓞⒖枷嚓P命令的man手冊。如man cat中有一行:
With no FILE, or when FILE is -, read standard input.
輸入重定向是<,除此之外還有<<、<<<。
<<符號表示here doc。也就是說,它后面跟的是一篇文檔,就像一個文件一樣,只不過這個文件的內容是臨時定義在<<符號后的。here doc常用于指定多行數據輸入。
既然是文檔,就有文檔起始符號表示文檔從此開始和文檔終止符號表示文檔到此結束。起始符和終止符中間的內容全部是文檔內容。文檔內容會作為標準輸入的數據被讀取。
文檔的起始符和終止符可以隨意定義,但兩者前后必須一樣。常見的符號是:
例如:
# here doc作為標準輸入被讀取,然后被cat輸出
cat <<EOF
hello
world
EOF
# here doc的內容還會被cat覆蓋式輸出到指定文件
cat <<eof >/tmp/file
hello
world
eof
# here doc的內容還會被cat追加式輸出到指定文件
cat <<eof >>/tmp/file
hello
world
eof
# here doc和重定向符號的前后位置隨意
cat >>/tmp/file<<eof
...
eof
另外,如果將起始符用引號包圍,則不會進行變量替換、命令替換、算術替換等。如果不用引號包圍起始符,則會進行替換。
a=333
cat <<eof
$a
eof
cat <<"eof"
$a
eof
輸出結果:
333
$a
<<<表示here string。也就是說該符號后面是一個字符串,這個字符串會作為標準輸入的內容。
cat <<<"www.junmajinlong.com"
使用單引號包圍here string時,不會進行變量替換、命令替換等,使用雙引號包圍時會進行替換。
$ a=3333
$ cat <<<$a
3333
$ cat <<<"hello world$a"
hello world3333
$ cat <<<'hello world$a'
hello world$a
here string??梢蕴娲艿狼暗膃cho命令echo xxx|
。例如:
# 下面是等價的
echo hello world | grep "llo"
grep "llo" <<<"hello world"
管道的用法:
cmd1 | cmd2 | cmd3...
每個豎線代表一個管道。上面命令行表示cmd1的標準輸出會放進管道,cmd2會從管道中讀取進行處理,cmd2的標準輸出會放入另一個管道,cmd3會從這個管道中讀取數據進行處理。后面還可以接任意數量的管道。
Shell管道是Shell中最值得稱贊的功能之一,它以非常簡潔的形式實現了管道的進程間通信方式,我個人認為Shell處理文本數據的半壁江山都來自于豎線形式的管道。像其它編程語言,打開管道后還要區分哪個進程寫管道、哪個進程讀管道,為了安全,每個進程還要關閉不用的讀端或寫端,總之就是麻煩,而Shell的管道非常簡潔,豎線左邊的就是寫管道的,豎線右邊的就是讀管道的。
例如:
ps aux | grep 'sshd'
ps命令產生的數據(標準輸出)會寫進管道,只要管道內一有數據,grep命令就從中讀取數據進行處理。
那下面的命令中,grep從哪讀數據呢?
ps aux | grep '#' /etc/fstab
那想要讓grep既從/etc/fstab讀取數據,也從管道中讀取數據呢?
ps aux | grep '#' /etc/fstab /etc/stdin
ps aux | grep '#' /etc/fstab -
tee命令可將一份標準輸入原樣拷貝到標準輸出和0或多個文件中。換句話說,tee的作用是數據多重定向。
NAME
tee - read from standard input and write to standard output and files
SYNOPSIS
tee [OPTION]... [FILE]...
DESCRIPTION
Copy standard input to each FILE, and also to standard output.
-a, --append
ppend to the given FILEs, do not overwrite
如圖:
例如:
$ echo hello world | tee /tmp/file1 /tmp/file2 | cat
$ echo hello world | tee -a /tmp/file3 >/dev/null
Bash還支持進程替換(注:有些Shell不支持進程替換)。
進程替換的語法:
<(cmd)
>(cmd)
進程替換和命令替換類似,都是讓cmd命令先執行,因為它們都是在Shell解析命令行的階段執行的。
進程替換先讓cmd放入后臺異步執行,并且不會等待cmd執行完。
其實,每個進程替換都是一個虛擬文件,只不過這個文件的內容是由cmd命令產生的(<(cmd)
)或被cmd命令讀取的(>(cmd)
)。
$ echo <(echo www.junmajinlong.com)
/dev/fd/63
既然進程替換是文件,那么它就可以像文件一樣被操作。比如被讀取、被當作標準輸入重定向的數據源等等:
# cmd做數據產生者
$ cat <(echo www.junmajinlong.com) # 等價于cat /dev/fd/63
$ cat < <(echo www.junmajinlong.com) # 等價于cat </dev/fd/63
# cmd做數據接收者
$ echo hello world > >(grep 'llo')
$ echo hello world | tee >(grep 'llo') >(grep 'rld') >/dev/null
test命令或功能等價的Bash內置命令[ ]
可以做條件測試,如果測試的結果為True,則退出狀態碼為0。
此外,還可以使用[[]]
來做條件測試,甚至let、$[]、$(())
也可以做條件測試,但這里暫不介紹。
這些條件測試常用在if、while語句中,也常用在cmd1 && cmd2 || cmd3
格式的命令行中。
用法示例:
sh_file=test.sh
[ -x "$sh_file" ] && ./$sh_file || { echo "can't execute,exit...";exit 1; }
test -x "$sh_file" && ./$sh_file || { echo "can't execute,exit...";exit 1; }
[]
中的條件測試需要和開閉中括號使用空格隔開,否則語法解析錯誤。
[ ]
test
沒有任何測試內容時,直接返回false。
true命令直接返回true,即退出狀態碼為0。
false命令直接返回false,即退出狀態碼非0。
$ true
$ echo $? # 0
$ false
$ echo $? # 1
條件表達式 | 含義 |
---|---|
-e file | 文件是否存在(exist) |
-f file | 文件是否存在且為普通文件(file) |
-d file | 文件是否存在且為目錄(directory) |
-b file | 文件是否存在且為塊設備block device |
-c file | 文件是否存在且為字符設備character device |
-S file | 文件是否存在且為套接字文件Socket |
-p file | 文件是否存在且為命名管道文件FIFO(pipe) |
-L file | 文件是否存在且是一個鏈接文件(Link) |
條件表達式 | 含義 |
---|---|
-r file | 文件是否存在且當前用戶可讀 |
-w file | 文件是否存在且當前用戶可寫 |
-x file | 文件是否存在且當前用戶可執行 |
-s file | 文件是否存在且大小大于0字節,即檢測文件是否非空文件 |
-N file | 文件是否存在,且自上次read后是否被modify |
條件表達式 | 含義 |
---|---|
file1 -nt file2 | (newer than)判斷file1是否比file2新 |
file1 -ot file2 | (older than)判斷file1是否比file2舊 |
file1 -ef file2 | (equal file)判斷file1與file2是否為同一文件 |
條件表達式 | 含義 |
---|---|
int1 -eq int2 | 兩數值相等(equal) |
int1 -ne int2 | 兩數值不等(not equal) |
int1 -gt int2 | n1大于n2(greater than) |
int1 -lt int2 | n1小于n2(less than) |
int1 -ge int2 | n1大于等于n2(greater than or equal) |
int1 -le int2 | n1小于等于n2(less than or equal) |
條件表達式 | 含義 |
---|---|
-z str | (zero)判定字符串是否為空?str為空串,則true |
str <br>-n str | 判定字符串是否非空?str為串,則false。注:-n可省略 |
str1 = str2 <br>str1 == str2 | str1和str2是否相同,相同則返回true。"=="和"="等價 |
str1 != str2 | str1是否不等于str2,若不等,則返回true |
str1 > str2 | str1字母順序是否大于str2,若大于則返回true |
str1 < str2 | str1字母順序是否小于str2,若小于則返回true |
條件表達式 | 含義 |
---|---|
-a或&& | (and)兩表達式同時為true時才為true。<br>"-a"只能在test或[]中使用,&&只能在[[]]中使用 |
-o或|| | (or)兩表達式任何一個true則為true。<br/>"-o"只能在test或[]中使用,||只能在[[]]中使用 |
! | 對表達式取反 |
( ) | 改變表達式的優先級,為了防止被shell解析,應加上反斜線轉義( ) |
if test-commands; then
consequent-commands;
[elif more-test-commands; then
more-consequents;]
[else alternate-consequents;]
fi
test-commands既可以是test測試或[]、[[]]
測試,也可以是任何其它命令,test-commands用于條件測試,它只判斷命令的退出狀態碼是否為0,為0則為true。
例如:
if [ "$a" ];then echo '$a' is not none;else echo '$a' undefined or empty;fi
if [ ! -d ~/.ssh ];then
mkdir ~/.ssh
chown -R $USER.$USER ~/.ssh
chmod 700 ~/.ssh
fi
if grep 'junmajinlong' /etc/passwd &>/dev/null;then
echo 'User "junmajinlong" already exists...'
elif grep 'malongshuai' /etc/passwd &>/dev/null;then
echo 'User "malongshuai" already exists...'
else
echo 'you should create user,exit...'
exit 1
fi
case常用于確定的分支判斷。比如:
while [ ${#@} -gt 0 ];do
case "$1" in
start)
echo start;;
stop)
echo stop
;;
restart)
echo restart
;;
reload | force-reload)
echo reload;;
status)
echo status;;
*)
echo $"Usage: $0 {start|stop|status|restart|reload|force-reload}"
exit 2
esac
done
case用法基本要求:
;;
結尾,否則出現分支穿透(所以;;
不是必須的)*)
下面是為ping命令自定義設計選項的腳本:
#!/bin/bash
while [ $1 ];do
case "$1" in
-c|--count)
count=$2
shift 2
;;
-t|--timeout)
timeout=$2
shift 2
;;
-h|--host)
host=$2
shift 2
;;
*)
echo "wrong options or arguments"
exit 1
esac
done
ping -c $count -W timeout $host
執行:
$ chmod +x ping.sh
$ ./ping.sh -c 5 -t 2
有兩種for循環結構:
# 成員測試類語法
for i in word1 word2 ...;do cmd_list;done
# C語言for語法
for (( expr1;expr2;expr3 ));do cmd_list;done
成員測試類的for循環中,in關鍵字后是使用空格分隔的一個或多個元素,for循環時,每次從in關鍵字后面取一個元素并賦值給i變量。
例如:
$ for i in 1 2 3 4;do echo $i;done
1
2
3
4
$ for i in 1 2 "3 4";do echo $i;done
1
2
3 4
C語言型的for語法中,expr1是初始化語句,expr2是循環終點條件判斷語句,expr3是每輪循環后執行的語句,一般用來更改條件判斷相關的變量。
for ((i=1;i<=3;++i));do echo $i;done
1
2
3
while test_cmd_list;do cmd_list;done
while循環,開始時會測試test_cmd_list
,如果測試的退出狀態碼為0,則執行一次循環體語句cmd_list
,然后再測試test_cmd_list
,一直循環,直到測試退出狀態碼非0,循環退出。
例如:
let i=1,sum=0;
while [ $i -le 10 ];do
let sum=sum+i
let ++i
done
還有until循環語句,但在Shell中用的很少。
while循環經常會和read命令一起使用,read是Bash的內置命令,可用來讀取文件,通常會按行讀?。好看巫x一行。
例如:
cat /etc/fstab | while read line;do
let num+=1
echo $num: $line
done
上面的命令行中,首先cat進程和while結構開始運行,while結構中的read命令從標準輸入中讀取,也就是從管道中讀取數據,每次讀取一行,因為管道中最初沒有數據,所以read命令被阻塞處于數據等待狀態。當cat命令讀完文件所有數據后,將數據放入到管道中,于是read命令從管道中每次讀取一行并將所讀行賦值給變量line,然后執行循環體,然后繼續循環,直到read讀完所有數據,循環退出。
但注意,管道兩邊的命令默認是在子Shell中執行的,所以其設置的變量在命令執行完成后就消失。換句話說,在父Shell中無法訪問這些變量。比如上面的num變量是在管道的while結構中設置的,除了在while中能訪問該變量,其它任何地方都無法訪問它。
如果想要訪問while中賦值的變量,就不能使用管道。如果是直接從文件讀取,可使用輸入重定向,如果是讀取命令產生的數據,可使用進程替換。
while read line;do
let num1+=1
echo $num1: $line
done </etc/fstab
echo $num1
while read line;do
let num2+=1
echo $num2: $line
done < <(grep 'UUID' /etc/fstab)
Shell函數可以當作命令一樣執行,它是一個或多個命令的組合結構體。通常,可以為每個功能定義一個函數,該函數中包含實現這個功能相關的所有命令和邏輯。
因為可以組合多個命令,并且定義之后就可以直接在當前Shell中調用,所以函數具有一次定義多次調用且代碼復用的功能。
Shell函數的定義風格有下面幾種:
function func_name {CMD_LIST}
func_name() {CMD_LIST}
function func_name() {CMD_LIST}
函數定義后,可以直接使用函數名來調用函數,同時還可以向函數傳遞零個或多個參數。
# 不傳遞參數
func_name
# 傳遞多個參數
func_name arg1 arg2 arg3
在函數中,那些位置變量將有特殊的意義:
$1、$2、$3...
:傳遞給函數的第一個參數保存在$1
中,第二個參數保存在$2
中...$@
和$*
:保存了所有參數,各參數使用空格分隔
$@
的各個元素都被雙引號包圍,$*
的所有元素一次性被雙引號包圍例如,定義一個函數專門用來設置和代理相關的變量:
proxy_addr=127.0.0.1:8118
function proxy_set {
local p_addr=$1
export http_proxy=$p_addr
export https_proxy=$p_addr
export ftp_proxy=$p_addr
}
# 調用函數
proxy_set $proxy_addr
# 各代理變量已設置
echo $http_proxy
echo $https_proxy
echo $ftp_proxy
上面在函數定義的代碼中使用了local
,它可以用在函數內部表示定義一個局部變量,局部變量在函數執行完畢后就消失,不會影響函數外部的環境。
另外,函數中可以使用return語句來定義函數的返回值,每當執行到函數內的return時,函數就會終止執行,直接退出函數。在Shell中,函數的返回值其實就是退出狀態碼。
return [N]
如果不指定N,則默認退出狀態碼為0。
例如:
function sayhello {
[ "$1" ] || { echo "give me a name please!"; return 1; }
echo hello $1
}
sayhello "junma" ; echo $?
sayhello "jinlong" ; echo $?
sayhello ; echo $?
2020-01-28
powershell遠程管理服務器磁盤空間的實現代碼2020-01-28
詳談Ubuntu PowerShell(小白入門必看教程)2020-01-28
Powershell 腳本數字簽名實現方法微軟發布了最新的powershell for sql server 2016命令行客戶端庫。文章介紹了與之相關的實用方法,需要的朋友可以參考下...
2020-01-28