原文連接https://www.jb51.net/article/161028.htmphp
什麼是Shell腳本java
Shell腳本(英語:Shell script),又稱Shell命令稿、程序化腳本,是一種電腦程序與文本文件,內容由一連串的shell命令組成,經由Unix Shell直譯其內容後運做。被當成是一種腳本語言來設計,其運做方式與直譯語言至關,由Unix shell扮演命令行解釋器的角色,在讀取shell腳本以後,依序運行其中的shell命令,以後輸出結果。利用shell腳本能夠進行系統管理,文件操做等。python
示例linux
看個例子吧:ios
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
#!/bin/sh
cd
~
mkdir
shell_tut
cd
shell_tut
for
((i=0; i<10; i++));
do
touch
test_$i.txt
done
|
示例解釋 •第1行:指定腳本解釋器,這裏是用/bin/sh作解釋器的 •第2行:切換到當前用戶的home目錄 •第3行:建立一個目錄shell_tut •第4行:切換到shell_tut目錄 •第5行:循環條件,一共循環10次 •第6行:建立一個test_1…10.txt文件 •第7行:循環體結束git
cd, mkdir, touch都是系統自帶的程序,通常在/bin或者/usr/bin目錄下。for, do, done是sh腳本語言的關鍵字。github
shell和shell腳本的概念web
shell是指一種應用程序,這個應用程序提供了一個界面,用戶經過這個界面訪問操做系統內核的服務。Ken Thompson的sh是第一種Unix Shell,Windows Explorer是一個典型的圖形界面Shell。shell
shell腳本(shell script),是一種爲shell編寫的腳本程序。業界所說的shell一般都是指shell腳本,但讀者朋友要知道,shell和shell script是兩個不一樣的概念。因爲習慣的緣由,簡潔起見,本文出現的「shell編程」都是指shell腳本編程,不是指開發shell自身(如Windows Explorer擴展開發)。編程
環境
shell編程跟java、php編程同樣,只要有一個能編寫代碼的文本編輯器和一個能解釋執行的腳本解釋器就能夠了。
OS
當前主流的操做系統都支持shell編程,本文檔所述的shell編程是指Linux下的shell,講的基本都是POSIX標準下的功能,因此,也適用於Unix及BSD(如Mac OS)。
Linux
Linux默認安裝就帶了shell解釋器。
Mac OS
Mac OS不只帶了sh、bash這兩個最基礎的解釋器,還內置了ksh、csh、zsh等不經常使用的解釋器。
Windows上的模擬器
windows出廠時沒有內置shell解釋器,須要自行安裝,爲了同時能用grep, awk, curl等工具,最好裝一個cygwin或者mingw來模擬linux環境。 •cygwin •mingw
腳本解釋器
sh
即Bourne shell,POSIX(Portable Operating System Interface)標準的shell解釋器,它的二進制文件路徑一般是/bin/sh,由Bell Labs開發。
本文講的是sh,若是你使用其它語言用做shell編程,請自行參考相應語言的文檔。
bash
Bash是Bourne shell的替代品,屬GNU Project,二進制文件路徑一般是/bin/bash。業界一般混用bash、sh、和shell,好比你會常常在招聘運維工程師的文案中見到:熟悉Linux Bash編程,精通Shell編程。
在CentOS裏,/bin/sh是一個指向/bin/bash的符號連接:
1
2
3
4
5
6
7
|
[root@centosraw ~]
# ls -l /bin/*sh
-rwxr-xr-x. 1 root root 903272 Feb 22 05:09
/bin/bash
-rwxr-xr-x. 1 root root 106216 Oct 17 2012
/bin/dash
lrwxrwxrwx. 1 root root 4 Mar 22 10:22
/bin/sh
->
bash
|
但在Mac OS上不是,/bin/sh和/bin/bash是兩個不一樣的文件,儘管它們的大小隻相差100字節左右:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
iMac:~ wuxiao$
ls
-l
/bin/
*sh
-r-xr-xr-x 1 root wheel 1371648 6 Nov 16:52
/bin/bash
-rwxr-xr-x 2 root wheel 772992 6 Nov 16:52
/bin/csh
-r-xr-xr-x 1 root wheel 2180736 6 Nov 16:52
/bin/ksh
-r-xr-xr-x 1 root wheel 1371712 6 Nov 16:52
/bin/sh
-rwxr-xr-x 2 root wheel 772992 6 Nov 16:52
/bin/tcsh
-rwxr-xr-x 1 root wheel 1103984 6 Nov 16:52
/bin/zsh
|
高級編程語言
理論上講,只要一門語言提供瞭解釋器(而不只是編譯器),這門語言就能夠勝任腳本編程,常見的解釋型語言都是能夠用做腳本編程的,如:Perl、Tcl、Python、PHP、Ruby。Perl是最老牌的腳本編程語言了,Python這些年也成了一些linux發行版的預置解釋器。
編譯型語言,只要有解釋器,也能夠用做腳本編程,如C shell是內置的(/bin/csh),Java有第三方解釋器Jshell,Ada有收費的解釋器AdaScript。
以下是一個PHP Shell Script示例(假設文件名叫test.php):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
#!/usr/bin/php
<?php
for
(
$i
=0;
$i
< 10;
$i
++)
echo
$i
.
"\n"
;
執行:
/usr/bin/php test.php
或者:
chmod
+x test.php
./test.php
|
如何選擇shell編程語言
熟悉 vs 陌生
若是你已經掌握了一門編程語言(如PHP、Python、Java、JavaScript),建議你就直接使用這門語言編寫腳本程序,雖然某些地方會有點囉嗦,但你能利用在這門語言領域裏的經驗(單元測試、單步調試、IDE、第三方類庫)。
新增的學習成本很小,只要學會怎麼使用shell解釋器(Jshell、AdaScript)就能夠了。
簡單 vs 高級
若是你以爲本身熟悉的語言(如Java、C)寫shell腳本實在太囉嗦,你只是想作一些備份文件、安裝軟件、下載數據之類的事情,學着使用sh,bash會是一個好主意。
shell只定義了一個很是簡單的編程語言,因此,若是你的腳本程序複雜度較高,或者要操做的數據結構比較複雜,那麼仍是應該使用Python、Perl這樣的腳本語言,或者是你原本就已經很擅長的高級語言。由於sh和bash在這方面很弱,好比說: •·它的函數只能返回字串,沒法返回數組 •·它不支持面向對象,你沒法實現一些優雅的設計模式 •·它是解釋型的,一邊解釋一邊執行,連PHP那種預編譯都不是,若是你的腳本包含錯誤(例如調用了不存在的函數),只要沒執行到這一行,就不會報錯
環境兼容性
若是你的腳本是提供給別的用戶使用,使用sh或者bash,你的腳本將具備最好的環境兼容性,perl很早就是linux標配了,python這些年也成了一些linux發行版的標配,至於mac os,它默認安裝了perl、python、ruby、php、java等主流編程語言。
第一個shell腳本
編寫
打開文本編輯器,新建一個文件,擴展名爲sh(sh表明shell),擴展名並不影響腳本執行,見名知意就好,若是你用php寫shell 腳本,擴展名就用php好了。
輸入一些代碼,第一行通常是這樣:
1
2
3
4
5
|
#!/bin/bash
#!/usr/bin/php
「
#!」是一個約定的標記,它告訴系統這個腳本須要什麼解釋器來執行。
|
運行
運行Shell腳本有兩種方法:
做爲可執行程序
1
2
|
chmod
+x
test
.sh
.
/test
.sh
|
注意,必定要寫成./test.sh,而不是test.sh,運行其它二進制的程序也同樣,直接寫test.sh,linux系統會去PATH裏尋找有沒有叫test.sh的,而只有/bin, /sbin, /usr/bin,/usr/sbin等在PATH裏,你的當前目錄一般不在PATH裏,因此寫成test.sh是會找不到命令的,要用./test.sh告訴系統說,就在當前目錄找。
經過這種方式運行bash腳本,第一行必定要寫對,好讓系統查找到正確的解釋器。
這裏的"系統",其實就是shell這個應用程序(想象一下Windows Explorer),但我故意寫成系統,是方便理解,既然這個系統就是指shell,那麼一個使用/bin/sh做爲解釋器的腳本是否是能夠省去第一行呢?是的。
做爲解釋器參數
這種運行方式是,直接運行解釋器,其參數就是shell腳本的文件名,如:
1
2
3
|
/bin/sh
test
.sh
/bin/php
test
.php
|
這種方式運行的腳本,不須要在第一行指定解釋器信息,寫了也沒用。
變量
定義變量
定義變量時,變量名不加美圓符號($),如:
1
|
your_name=
"qinjx"
|
注意,變量名和等號之間不能有空格,這可能和你熟悉的全部編程語言都不同。
除了顯式地直接賦值,還能夠用語句給變量賦值,如:
1
|
for
file
in
`
ls
/etc
`
|
使用變量
使用一個定義過的變量,只要在變量名前面加美圓符號便可,如:
1
2
3
4
5
|
your_name=
"qinjx"
echo
$your_name
echo
${your_name}
|
變量名外面的花括號是可選的,加不加都行,加花括號是爲了幫助解釋器識別變量的邊界,好比下面這種狀況:
1
2
3
4
5
|
for
skill
in
Ada Coffe Action Java;
do
echo
"I am good at ${skill}Script"
done
|
若是不給skill變量加花括號,寫成echo "I am good at $skillScript",解釋器就會把$skillScript當成一個變量(其值爲空),代碼執行結果就不是咱們指望的樣子了。
推薦給全部變量加上花括號,這是個好的編程習慣。IntelliJ IDEA編寫shell script時,IDE就會提示加花括號。
重定義變量
已定義的變量,能夠被從新定義,如:
1
2
3
4
5
6
7
|
your_name=
"qinjx"
echo
$your_name
your_name=
"alibaba"
echo
$your_name
|
這樣寫是合法的,但注意,第二次賦值的時候不能寫$your_name="alibaba",使用變量的時候才加美圓符。
註釋
以「#」開頭的行就是註釋,會被解釋器忽略。
多行註釋
sh裏沒有多行註釋,只能每一行加一個#號。就像這樣:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
#--------------------------------------------
# 這是一個自動打ipa的腳本,基於webfrogs的ipa-build書寫:https://github.com/webfrogs/xcode_shell/blob/master/ipa-build
# 功能:自動爲etao ios app打包,產出物爲14個渠道的ipa包
# 特點:全自動打包,不須要輸入任何參數
#--------------------------------------------
##### 用戶配置區 開始 #####
#
#
# 項目根目錄,推薦將此腳本放在項目的根目錄,這裏就不用改了
# 應用名,確保和Xcode裏Product下的target_name.app名字一致
#
##### 用戶配置區 結束 #####
|
若是在開發過程當中,遇到大段的代碼須要臨時註釋起來,過一下子又取消註釋,怎麼辦呢?每一行加個#符號太費力了,能夠把這一段要註釋的代碼用一對花括號括起來,定義成一個函數,沒有地方調用這個函數,這塊代碼就不會執行,達到了和註釋同樣的效果。
字符串
字符串是shell編程中最經常使用最有用的數據類型(除了數字和字符串,也沒啥其它類型好用了,哈哈),字符串能夠用單引號,也能夠用雙引號,也能夠不用引號。單雙引號的區別跟PHP相似。
單引號
1
|
str=
'this is a string'
|
單引號字符串的限制: •單引號裏的任何字符都會原樣輸出,單引號字符串中的變量是無效的 •單引號字串中不能出現單引號(對單引號使用轉義符後也不行)
雙引號
1
|
str=
"Hello, I know your are \"$your_name\"! \n"
|
•雙引號裏能夠有變量 •雙引號裏能夠出現轉義字符
字符串操做
拼接字符串
1
2
3
4
5
6
7
8
9
|
your_name=
"qinjx"
greeting=
"hello, "
$your_name
" !"
greeting_1=
"hello, ${your_name} !"
echo
$greeting $greeting_1
|
獲取字符串長度:
1
2
|
string=
"abcd"
echo
${
#string} #輸出:4
|
提取子字符串
1
2
3
|
string=
"alibaba is a great company"
echo
${string:1:4}
#輸出:liba
|
查找子字符串
1
2
3
|
string=
"alibaba is a great company"
echo
`
expr
index
"$string"
is`
#輸出:8,這個語句的意思是:找出單詞is在這名話中的位置
|
數組
管道
條件判斷
流程控制
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
|
和Java、PHP等語言不同,sh的流程控制不可爲空,如:
<?php
if
(isset($_GET[
"q"
])) {
search(q);
}
else
{
//do
nothing
}
在sh
/bash
裏可不能這麼寫,若是
else
分支沒有語句執行,就不要寫這個
else
。
還要注意,sh裏的
if
[ $foo -
eq
0 ],這個方括號跟Java
/PHP
裏
if
後面的圓括號大不相同,它是一個可執行程序(和
cd
,
ls
,
grep
同樣),想不到吧?在CentOS上,它在
/usr/bin
目錄下:
ll
/usr/bin/
[
-rwxr-xr-x. 1 root root 33408 6月 22 2012
/usr/bin/
[
正由於方括號在這裏是一個可執行程序,方括號後面必須加空格,不能寫成
if
[$foo -
eq
0]
if
else
if
if
condition
then
command1
command2
...
commandN
fi
寫成一行(適用於終端命令提示符):
if
`
ps
-ef |
grep
ssh
`;
then
echo
hello;
fi
末尾的
fi
就是
if
倒過來拼寫,後面還會遇到相似的
if
else
if
condition
then
command1
command2
...
commandN
else
command
fi
if
else
-
if
else
if
condition1
then
command1
elif
condition2
command2
else
commandN
fi
for
while
for
在開篇的示例裏演示過了:
for
var
in
item1 item2 ... itemN
do
command1
command2
...
commandN
done
寫成一行:
for
var
in
item1 item2 ... itemN;
do
command1; command2…
done
;
|
C風格的for
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
|
for
(( EXP1; EXP2; EXP3 ))
do
command1
command2
command3
done
while
while
condition
do
command
done
無限循環
while
:
do
command
done
或者
while
true
do
command
done
或者
for
(( ; ; ))
until
until
condition
do
command
done
case
case
"${opt}"
in
"Install-Puppet-Server"
)
install_master $1
exit
;;
"Install-Puppet-Client"
)
install_client $1
exit
;;
"Config-Puppet-Server"
)
config_puppet_master
exit
;;
"Config-Puppet-Client"
)
config_puppet_client
exit
;;
"Exit"
)
exit
;;
* )
echo
"Bad option, please choose again"
esac
case
的語法和C family語言差異很大,它須要一個esac(就是
case
反過來)做爲結束標記,每一個
case
分支用右圓括號,用兩個分號表示
break
|
函數
定義
調用
文件包含
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
可使用
source
和.關鍵字,如:
source
.
/function
.sh
. .
/function
.sh
在
bash
裏,
source
和.是等效的,他們都是讀入
function
.sh的內容並執行其內容(相似PHP裏的include),爲了更好的可移植性,推薦使用第二種寫法。
包含一個文件和執行一個文件同樣,也要寫這個文件的路徑,不能光寫文件名,好比上述例子中:
. .
/function
.sh
不能夠寫做:
.
function
.sh
若是
function
.sh是用戶傳入的參數,如何得到它的絕對路徑呢?方法是:
real_path=`readlink -f $1`
#$1是用戶輸入的參數,如function.sh
. $real_path
|
用戶輸入
執行腳本時傳入
腳本運行中輸入
select菜單
stdin和stdout
經常使用的命令
sh腳本結合系統命令便有了強大的威力,在字符處理領域,有grep、awk、sed三劍客,grep負責找出特定的行,awk能將行拆分紅多個字段,sed則能夠實現更新插入刪除等寫操做。
ps
查看進程列表
grep
排除grep自身
查找與target相鄰的結果
以上就是本文的所有內容,但願對你們的學習有所幫助,也但願你們多多支持腳本之家。