成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專(zhuān)欄INFORMATION COLUMN

玩轉(zhuǎn)Linux - shell 編程基礎(chǔ)

null1145 / 1417人閱讀

摘要:數(shù)組中可以存放多個(gè)值,與大部分編程語(yǔ)言類(lèi)似,數(shù)組元素的下標(biāo)由開(kāi)始。定義數(shù)組在中,用括號(hào)來(lái)表示數(shù)組,數(shù)組元素用空格符號(hào)分割開(kāi)。

簡(jiǎn)述

使用 linux 就離不開(kāi) shell,那么也就是說(shuō)也離不開(kāi) shell 編程。很多時(shí)候服務(wù)器都需要編寫(xiě)一些計(jì)劃任務(wù)來(lái)定時(shí)運(yùn)行的,所以掌握一些基本的 shell 編程基礎(chǔ)很有必要。

本文是我在網(wǎng)上收集的一些資料,主要目的是幫助自己更好的了解掌握 shell 編程的一些基礎(chǔ)知識(shí)。

什么是Shell腳本

示例看個(gè)例子吧:

#!/bin/sh
cd ~
mkdir shell_tut
cd shell_tut

for ((i=0; i<10; i++)); do
    touch test_$i.txt
done

示例解釋?zhuān)?/p>

第1行:指定腳本解釋器,這里是用/bin/sh做解釋器的

第2行:切換到當(dāng)前用戶(hù)的home目錄

第3行:創(chuàng)建一個(gè)目錄shell_tut

第4行:切換到shell_tut目錄

第5行:循環(huán)條件,一共循環(huán)10次

第6行:創(chuàng)建一個(gè)test_1…10.txt文件

第7行:循環(huán)體結(jié)束

cd, mkdir, touch 都是系統(tǒng)自帶的程序,一般在/bin或者/usr/bin目錄下。
for, do, done 是sh腳本語(yǔ)言的關(guān)鍵字。
shell和shell腳本的概念

shell是指一種應(yīng)用程序,這個(gè)應(yīng)用程序提供了一個(gè)界面,用戶(hù)通過(guò)這個(gè)界面訪(fǎng)問(wèn)操作系統(tǒng)內(nèi)核的服務(wù)。

Ken Thompson的sh是第一種Unix Shell,Windows Explorer是一個(gè)典型的圖形界面Shell。
shell腳本(shell script),是一種為shell編寫(xiě)的腳本程序。

業(yè)界所說(shuō)的shell通常都是指shell腳本,但讀者朋友要知道,shell和shell script是兩個(gè)不同的概念。

由于習(xí)慣的原因,簡(jiǎn)潔起見(jiàn),本文出現(xiàn)的“shell編程”都是指shell腳本編程,不是指開(kāi)發(fā)shell自身(如Windows Explorer擴(kuò)展開(kāi)發(fā))。

環(huán)境shell編程跟java、php編程一樣,只要有一個(gè)能編寫(xiě)代碼的文本編輯器和一個(gè)能解釋執(zhí)行的腳本解釋器就可以了。

OS當(dāng)前主流的操作系統(tǒng)都支持shell編程,本文檔所述的shell編程是指Linux下的shell,講的基本都是POSIX標(biāo)準(zhǔn)下的功能,所以,也適用于Unix及BSD(如Mac OS)。

LinuxLinux默認(rèn)安裝就帶了shell解釋器。

Mac OSMac OS不僅帶了sh、bash這兩個(gè)最基礎(chǔ)的解釋器,還內(nèi)置了ksh、csh、zsh等不常用的解釋器。

Windows上的模擬器windows出廠(chǎng)時(shí)沒(méi)有內(nèi)置shell解釋器,需要自行安裝,為了同時(shí)能用grep, awk, curl等工具,最好裝一個(gè)cygwin或者mingw來(lái)模擬linux環(huán)境。

cygwin

mingw

腳本解釋器 sh

即Bourne shell,POSIX(Portable Operating System Interface)標(biāo)準(zhǔn)的shell解釋器,它的二進(jìn)制文件路徑通常是/bin/sh,由Bell Labs開(kāi)發(fā)。

bashBash是Bourne shell的替代品,屬GNU Project,二進(jìn)制文件路徑通常是/bin/bash。業(yè)界通?;煊胋ash、sh、和shell,比如你會(huì)經(jīng)常在招聘運(yùn)維工程師的文案中見(jiàn)到:熟悉Linux Bash編程,精通Shell編程。

在CentOS里,/bin/sh是一個(gè)指向/bin/bash的符號(hào)鏈接:

[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是兩個(gè)不同的文件,盡管它們的大小只相差100字節(jié)左右:

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
如何選擇shell編程語(yǔ)言

熟悉 vs 陌生如果你已經(jīng)掌握了一門(mén)編程語(yǔ)言(如PHP、Python、Java、JavaScript),建議你就直接使用這門(mén)語(yǔ)言編寫(xiě)腳本程序,雖然某些地方會(huì)有點(diǎn)啰嗦,但你能利用在這門(mén)語(yǔ)言領(lǐng)域里的經(jīng)驗(yàn)(單元測(cè)試、單步調(diào)試、IDE、第三方類(lèi)庫(kù))。

新增的學(xué)習(xí)成本很小,只要學(xué)會(huì)怎么使用shell解釋器(Jshell、AdaScript)就可以了。

簡(jiǎn)單 vs 高級(jí)如果你覺(jué)得自己熟悉的語(yǔ)言(如Java、C)寫(xiě)shell腳本實(shí)在太啰嗦,你只是想做一些備份文件、安裝軟件、下載數(shù)據(jù)之類(lèi)的事情,學(xué)著使用sh,bash會(huì)是一個(gè)好主意。

shell只定義了一個(gè)非常簡(jiǎn)單的編程語(yǔ)言,所以,如果你的腳本程序復(fù)雜度較高,或者要操作的數(shù)據(jù)結(jié)構(gòu)比較復(fù)雜,那么還是應(yīng)該使用Python、Perl這樣的腳本語(yǔ)言,或者是你本來(lái)就已經(jīng)很擅長(zhǎng)的高級(jí)語(yǔ)言。

因?yàn)閟h和bash在這方面很弱,比如說(shuō):

它的函數(shù)只能返回字串,無(wú)法返回?cái)?shù)組

它不支持面向?qū)ο?,你無(wú)法實(shí)現(xiàn)一些優(yōu)雅的設(shè)計(jì)模式

它是解釋型的,一邊解釋一邊執(zhí)行,連PHP那種預(yù)編譯都不是,如果你的腳本包含錯(cuò)誤(例如調(diào)用了不存在的函數(shù)),只要沒(méi)執(zhí)行到這一行,就不會(huì)報(bào)錯(cuò)

環(huán)境兼容性如果你的腳本是提供給別的用戶(hù)使用,使用sh或者bash,你的腳本將具有最好的環(huán)境兼容性,perl很早就是linux標(biāo)配了,python這些年也成了一些linux發(fā)行版的標(biāo)配,至于mac os,它默認(rèn)安裝了perl、python、ruby、php、java等主流編程語(yǔ)言。

第一個(gè)shell腳本 編寫(xiě)打開(kāi)文本編輯器,新建一個(gè)文件,擴(kuò)展名為sh(sh代表shell),擴(kuò)展名并不影響腳本執(zhí)行,見(jiàn)名知意就好,如果你用php寫(xiě)shell 腳本,擴(kuò)展名就用php好了。

輸入一些代碼,第一行一般是這樣:

#!/bin/bash
#!/usr/bin/php

“#!”是一個(gè)約定的標(biāo)記,它告訴系統(tǒng)這個(gè)腳本需要什么解釋器來(lái)執(zhí)行。

運(yùn)行

運(yùn)行Shell腳本有兩種方法:

1、作為可執(zhí)行程序

將上面的代碼保存為test.sh,并cd到相應(yīng)目錄:

chmod +x ./test.sh  #使腳本具有執(zhí)行權(quán)限
./test.sh  #執(zhí)行腳本

注意:

一定要寫(xiě)成./test.sh,而不是test.sh,運(yùn)行其它二進(jìn)制的程序也一樣。

直接寫(xiě)test.sh,linux 系統(tǒng)會(huì)去PATH里尋找有沒(méi)有叫test.sh的,只有/bin, /sbin, /usr/bin,/usr/sbin等在PATH里。

你的當(dāng)前目錄通常不在PATH里,所以寫(xiě)成test.sh是會(huì)找不到命令的,要用./test.sh告訴系統(tǒng)說(shuō),就在當(dāng)前目錄找。

2、作為解釋器參數(shù)

這種運(yùn)行方式是,直接運(yùn)行解釋器,其參數(shù)就是shell腳本的文件名,如:

/bin/sh test.sh
/bin/php test.php

這種方式運(yùn)行的腳本,不需要在第一行指定解釋器信息,寫(xiě)了也沒(méi)用。

變量

定義變量定義變量時(shí),變量名不加美元符號(hào)($),如:

your_name="qinjx"

注意,變量名和等號(hào)之間不能有空格,這可能和你熟悉的所有編程語(yǔ)言都不一樣:

首個(gè)字母必須為字母(a-z,A-Z)。

中間不能有空格,可以使用下劃線(xiàn)。

不能使用表單符號(hào)。

不能使用bash里的關(guān)鍵字(如果不清楚,可以用help命令查看bash的保留關(guān)鍵字)。

除了顯式地直接賦值,還可以用語(yǔ)句給變量賦值,如:

for file in `ls /etc`

說(shuō)明:上面的命令的意思是實(shí)現(xiàn)將 /etc 下目錄的文件名循環(huán)出來(lái)。

使用變量

使用一個(gè)定義過(guò)得變量,只要在變量名之前加上美元符號(hào) $ 即可,比如:

your_name="qinjx"
echo $your_name
echo ${your_name}

變量名外面的花括號(hào)是可選的,可加可不加,加花括號(hào)是為了幫助解釋器識(shí)別變量的邊界,比如當(dāng)遇到下面的情況時(shí):

for skill in Ada Coffe Action Java; do
    echo "I am good at ${skill}Script"
done

如果不給skill變量加上花括號(hào),變量寫(xiě)成 $skillScript,那么解釋器就識(shí)別不出$skill了,那么代碼的最終結(jié)果就跟我的預(yù)期相差甚遠(yuǎn)。

推薦給所有變量加上花括號(hào),這是shell編程的好習(xí)慣。

對(duì)于已定義的變量,可以被重新定義,比如:

your_name="tom"
echo $your_name
your_name="alibaba"
echo $your_name

注意第二次賦值的時(shí)候不能寫(xiě)成:

$your_name="alibaba"

只有在使用變量的時(shí)候才加美元符 $。

只讀變量

使用 readonly 命令可以將變量定義為只讀變量,只讀變量的值不能被改變。

下面的例子嘗試更改只讀變量,結(jié)果報(bào)錯(cuò):

#!/bin/bash
myUrl="http://www.w3cschool.cc"
readonly myUrl
myUrl="http://www.runoob.com"

運(yùn)行腳本,結(jié)果如下:

/bin/sh: NAME: This variable is read only.

刪除變量

使用 unset 命令可以刪除變量。語(yǔ)法:

unset variable_name

變量被刪除后不能再次使用。unset 命令不能刪除只讀變量。

案例:

#!/bin/sh
myUrl="http://www.runoob.com"
unset myUrl
echo $myUrl

上面的程序執(zhí)行將沒(méi)有任何輸出。

變量類(lèi)型

運(yùn)行 shell 程序時(shí),會(huì)同時(shí)存在三種變量:

局部變量: 局部變量在腳本或命令中定義,僅在當(dāng)前shell實(shí)例中有效,其他shell啟動(dòng)的程序不能訪(fǎng)問(wèn)局部變量。

環(huán)境變量: 所有的程序,包括shell啟動(dòng)的程序,都能訪(fǎng)問(wèn)環(huán)境變量,有些程序需要環(huán)境變量來(lái)保證其正常運(yùn)行。必要的時(shí)候shell腳本也可以定義環(huán)境變量。

shell變量:shell變量是由shell程序設(shè)置的特殊變量。shell變量中有一部分是環(huán)境變量,有一部分是局部變量,這些變量保證了shell的正常運(yùn)行

字符串

字符串是 shell 編程中最常用最有用的數(shù)據(jù)類(lèi)型(除了數(shù)字和字符串,也沒(méi)啥其它類(lèi)型好用了)。

字符串可以用單引號(hào),也可以用雙引號(hào),也可以不用引號(hào)。單雙引號(hào)的區(qū)別跟PHP類(lèi)似。

單引號(hào)

str="this is a string"

單引號(hào)字符串的限制:

單引號(hào)里的任何字符都會(huì)原樣輸出,單引號(hào)字符串中的變量是無(wú)效的。

單引號(hào)字串中不能出現(xiàn)單引號(hào)(對(duì)單引號(hào)使用轉(zhuǎn)義符后也不行)。

雙引號(hào)

your_name="qinjx"
str="Hello, I know your are "$your_name"! 
"

雙引號(hào)的優(yōu)點(diǎn):

雙引號(hào)里可以有變量。

雙引號(hào)里可以出現(xiàn)轉(zhuǎn)義字符。

拼接字符串

your_name="qinjx"
greeting="hello, "$your_name" !"
greeting_1="hello, ${your_name} !"
echo $greeting $greeting_1

獲取字符串長(zhǎng)度

string="abcd"
echo ${#string} #輸出 4

提取子字符串

以下實(shí)例從字符串第 2 個(gè)字符開(kāi)始截取 4 個(gè)字符:

string="runoob is a great site"
echo ${string:1:4} # 輸出 unoo

查找子字符串

查找字符 "i 或 s" 的位置:

string="runoob is a great company"
echo `expr index "$string" is`  # 輸出 8

注意: 以上腳本中 "`" 是反引號(hào),而不是單引號(hào) """,千萬(wàn)不要看錯(cuò)了。

數(shù)組

Bash Shell 只支持一維數(shù)組(不支持多維數(shù)組),初始化時(shí)不需要定義數(shù)組大小(與 PHP 類(lèi)似)。

獲取數(shù)組中的元素要利用下標(biāo),下標(biāo)可以是整數(shù)或算術(shù)表達(dá)式,其值應(yīng)大于或等于0。

數(shù)組中可以存放多個(gè)值,與大部分編程語(yǔ)言類(lèi)似,數(shù)組元素的下標(biāo)由0開(kāi)始。

定義數(shù)組

在Shell中,用括號(hào)來(lái)表示數(shù)組,數(shù)組元素用"空格"符號(hào)分割開(kāi)。

定義數(shù)組的一般形式為:

數(shù)組名=(值1 值2 ... 值n)

例如:

array_name=(value0 value1 value2 value3)

或者

array_name=(
value0
value1
value2
value3
)

還可以多帶帶定義數(shù)組的各個(gè)分量:

array_name[0]=value0
array_name[1]=value1
array_name[n]=valuen

可以不使用連續(xù)的下標(biāo),而且下標(biāo)的范圍沒(méi)有限制。

讀取數(shù)組

讀取數(shù)組元素值的一般格式是:

${數(shù)組名[下標(biāo)]}

例如:

valuen=${array_name[n]}

shell 編程實(shí)戰(zhàn):

#!/bin/bash

my_array=(A B "C" D)

echo "第一個(gè)元素為: ${my_array[0]}"
echo "第二個(gè)元素為: ${my_array[1]}"
echo "第三個(gè)元素為: ${my_array[2]}"
echo "第四個(gè)元素為: ${my_array[3]}"

執(zhí)行腳本,輸出結(jié)果如下所示:

$ chmod +x test.sh 
$ ./test.sh
第一個(gè)元素為: A
第二個(gè)元素為: B
第三個(gè)元素為: C
第四個(gè)元素為: D

使用 @ 或者 * 符號(hào)可以獲取數(shù)組中的所有元素,例如:

echo ${array_name[@]}

shell 編程實(shí)戰(zhàn):

#!/bin/bash

my_array[0]=A
my_array[1]=B
my_array[2]=C
my_array[3]=D

echo "數(shù)組的元素為: ${my_array[*]}"
echo "數(shù)組的元素為: ${my_array[@]}"

執(zhí)行腳本,輸出結(jié)果如下所示:

$ chmod +x test.sh 
$ ./test.sh
數(shù)組的元素為: A B C D
數(shù)組的元素為: A B C D

獲取數(shù)組的長(zhǎng)度

獲取數(shù)組長(zhǎng)度的方法與獲取字符串長(zhǎng)度的方法相同,例如:

# 取得數(shù)組元素的個(gè)數(shù)
length=${#array_name[@]}
# 或者
length=${#array_name[*]}
# 取得數(shù)組單個(gè)元素的長(zhǎng)度
lengthn=${#array_name[n]}

shell 實(shí)戰(zhàn)編程:

#!/bin/bash

my_array[0]=A
my_array[1]=B
my_array[2]=C
my_array[3]=D

echo "數(shù)組元素個(gè)數(shù)為: ${#my_array[*]}"
echo "數(shù)組元素個(gè)數(shù)為: ${#my_array[@]}"

執(zhí)行腳本,輸出結(jié)果如下所示:

$ chmod +x test.sh 
$ ./test.sh
數(shù)組元素個(gè)數(shù)為: 4
數(shù)組元素個(gè)數(shù)為: 4
注釋

以"#"開(kāi)頭的行就是注釋?zhuān)瑫?huì)被解釋器忽略。

sh里沒(méi)有多行注釋?zhuān)荒苊恳恍屑右粋€(gè)#號(hào)。只能像這樣:

#--------------------------------------------
# 這是一個(gè)注釋
#--------------------------------------------
##### 用戶(hù)配置區(qū) 開(kāi)始 #####
#
#
# 這里可以添加腳本描述信息
# 
#
##### 用戶(hù)配置區(qū) 結(jié)束  #####

如果在開(kāi)發(fā)過(guò)程中,遇到大段的代碼需要臨時(shí)注釋起來(lái),過(guò)一會(huì)兒又取消注釋?zhuān)趺崔k呢?

每一行加個(gè)#符號(hào)太費(fèi)力了,可以把這一段要注釋的代碼用一對(duì)花括號(hào)括起來(lái),定義成一個(gè)函數(shù),沒(méi)有地方調(diào)用這個(gè)函數(shù),這塊代碼就不會(huì)執(zhí)行,達(dá)到了和注釋一樣的效果。

傳遞參數(shù)

我們可以在執(zhí)行 Shell 腳本時(shí),向腳本傳遞參數(shù),腳本內(nèi)獲取參數(shù)的格式為:$n。

n 代表一個(gè)數(shù)字,1 為執(zhí)行腳本的第一個(gè)參數(shù),2 為執(zhí)行腳本的第二個(gè)參數(shù),以此類(lèi)推……

實(shí)例
以下實(shí)例我們向腳本傳遞三個(gè)參數(shù),并分別輸出,其中 $0 為執(zhí)行的文件名:

#!/bin/bash

echo "Shell 傳遞參數(shù)實(shí)例!";
echo "執(zhí)行的文件名:$0";
echo "第一個(gè)參數(shù)為:$1";
echo "第二個(gè)參數(shù)為:$2";
echo "第三個(gè)參數(shù)為:$3";

為腳本設(shè)置可執(zhí)行權(quán)限,并執(zhí)行腳本,輸出結(jié)果如下所示:

$ chmod +x test.sh 
$ ./test.sh 1 2 3
Shell 傳遞參數(shù)實(shí)例!
執(zhí)行的文件名:test.sh
第一個(gè)參數(shù)為:1
第二個(gè)參數(shù)為:2
第三個(gè)參數(shù)為:3

另外,還有幾個(gè)特殊字符用來(lái)處理參數(shù):

參數(shù)處理    說(shuō)明
$#    傳遞到腳本的參數(shù)個(gè)數(shù)
$*    以一個(gè)單字符串顯示所有向腳本傳遞的參數(shù)。
如"$*"用「"」括起來(lái)的情況、以"$1 $2 … $n"的形式輸出所有參數(shù)。
$$    腳本運(yùn)行的當(dāng)前進(jìn)程ID號(hào)
$!    后臺(tái)運(yùn)行的最后一個(gè)進(jìn)程的ID號(hào)
$@    與$*相同,但是使用時(shí)加引號(hào),并在引號(hào)中返回每個(gè)參數(shù)。
如"$@"用「"」括起來(lái)的情況、以"$1" "$2" … "$n" 的形式輸出所有參數(shù)。
$-    顯示Shell使用的當(dāng)前選項(xiàng),與set命令功能相同。
$?    顯示最后命令的退出狀態(tài)。0表示沒(méi)有錯(cuò)誤,其他任何值表明有錯(cuò)誤。
#!/bin/bash

echo "Shell 傳遞參數(shù)實(shí)例!";
echo "第一個(gè)參數(shù)為:$1";

echo "參數(shù)個(gè)數(shù)為:$#";
echo "傳遞的參數(shù)作為一個(gè)字符串顯示:$*";

執(zhí)行腳本,輸出結(jié)果如下所示:

$ chmod +x test.sh 
$ ./test.sh 1 2 3
Shell 傳遞參數(shù)實(shí)例!
第一個(gè)參數(shù)為:1
參數(shù)個(gè)數(shù)為:3
傳遞的參數(shù)作為一個(gè)字符串顯示:1 2 3

$* 與 $@ 區(qū)別:

相同點(diǎn):都是引用所有參數(shù)。

不同點(diǎn):只有在雙引號(hào)中體現(xiàn)出來(lái)。

假設(shè)在腳本運(yùn)行時(shí)寫(xiě)了三個(gè)參數(shù) 1、2、3,,則 " * " 等價(jià)于 "1 2 3"(傳遞了一個(gè)參數(shù)),而 "@" 等價(jià)于 "1" "2" "3"(傳遞了三個(gè)參數(shù))。
#!/bin/bash

echo "-- $* 演示 ---"
for i in "$*"; do
    echo $i
done

echo "-- $@ 演示 ---"
for i in "$@"; do
    echo $i
done

執(zhí)行腳本,輸出結(jié)果如下所示:

$ chmod +x test.sh 
$ ./test.sh 1 2 3
-- $* 演示 ---
1 2 3
-- $@ 演示 ---
1
2
3
基本運(yùn)算符

Shell 和其他編程語(yǔ)言一樣,支持多種運(yùn)算符,包括:

算數(shù)運(yùn)算符

關(guān)系運(yùn)算符

布爾運(yùn)算符

字符串運(yùn)算符

文件測(cè)試運(yùn)算符

原生bash不支持簡(jiǎn)單的數(shù)學(xué)運(yùn)算,但是可以通過(guò)其他命令來(lái)實(shí)現(xiàn),例如 awk 和 expr,expr 最常用。

expr 是一款表達(dá)式計(jì)算工具,使用它能完成表達(dá)式的求值操作。

例如,兩個(gè)數(shù)相加(特別注意:使用的是反引號(hào) ` 而不是單引號(hào) "):

#!/bin/bash

val=`expr 2 + 2`
echo "兩數(shù)之和為 : $val"

執(zhí)行腳本,輸出結(jié)果如下所示:

兩數(shù)之和為 : 4

兩點(diǎn)注意:

表達(dá)式和運(yùn)算符之間要有空格,例如 2+2 是不對(duì)的,必須寫(xiě)成 2 + 2,這與我們熟悉的大多數(shù)編程語(yǔ)言不一樣。

完整的表達(dá)式要被 包含,注意這個(gè)字符不是常用的單引號(hào),在 Esc 鍵下邊。

算術(shù)運(yùn)算符

下面列出了常用的算術(shù)運(yùn)算符,假定變量 a 為 10,變量 b 為 20:

運(yùn)算符    說(shuō)明    舉例
 + 加法    `expr $a + $b` 結(jié)果為 30。
 + 減法    `expr $a - $b` 結(jié)果為 10。
 + 乘法    `expr $a * $b` 結(jié)果為  200。
/    除法    `expr $b / $a` 結(jié)果為 2。
%    取余    `expr $b % $a` 結(jié)果為 0。
=    賦值    a=$b 將把變量 b 的值賦給 a。
==    相等。用于比較兩個(gè)數(shù)字,相同則返回 true。    [ $a == $b ] 返回 false。
!=    不相等。用于比較兩個(gè)數(shù)字,不相同則返回 true。    [ $a != $b ] 返回 true。

注意:條件表達(dá)式要放在方括號(hào)之間,并且要有空格,例如: [$a==$b] 是錯(cuò)誤的,必須寫(xiě)成 [ $a == $b ]。

示例

算術(shù)運(yùn)算符實(shí)例如下:

#!/bin/bash

a=10
b=20

val=`expr $a + $b`
echo "a + b : $val"

val=`expr $a - $b`
echo "a - b : $val"

val=`expr $a * $b`
echo "a * b : $val"

val=`expr $b / $a`
echo "b / a : $val"

val=`expr $b % $a`
echo "b % a : $val"

if [ $a == $b ]
then
   echo "a 等于 b"
fi
if [ $a != $b ]
then
   echo "a 不等于 b"
fi

執(zhí)行腳本,輸出結(jié)果如下所示:

a + b : 30
a - b : -10
a * b : 200
b / a : 2
b % a : 0
a 不等于 b

注意:

乘號(hào)(*)前邊必須加反斜杠()才能實(shí)現(xiàn)乘法運(yùn)算。

if...then...fi 是條件語(yǔ)句,后續(xù)會(huì)有說(shuō)明。

關(guān)系運(yùn)算符

關(guān)系運(yùn)算符只支持?jǐn)?shù)字,不支持字符串,除非字符串的值是數(shù)字。

下面列出了常用的關(guān)系運(yùn)算符,假定變量 a 為 10,變量 b 為 20:

運(yùn)算符    說(shuō)明    舉例
-eq    檢測(cè)兩個(gè)數(shù)是否相等,相等返回 true。    [ $a -eq $b ] 返回 false。
-ne    檢測(cè)兩個(gè)數(shù)是否相等,不相等返回 true。    [ $a -ne $b ] 返回 true。
-gt    檢測(cè)左邊的數(shù)是否大于右邊的,如果是,則返回 true。    [ $a -gt $b ] 返回 false。
-lt    檢測(cè)左邊的數(shù)是否小于右邊的,如果是,則返回 true。    [ $a -lt $b ] 返回 true。
-ge    檢測(cè)左邊的數(shù)是否大等于右邊的,如果是,則返回 true。    [ $a -ge $b ] 返回 false。
-le    檢測(cè)左邊的數(shù)是否小于等于右邊的,如果是,則返回 true。    [ $a -le $b ] 返回 true。

實(shí)例

關(guān)系運(yùn)算符實(shí)例如下:

#!/bin/bash

a=10
b=20

if [ $a -eq $b ]
then
   echo "$a -eq $b : a 等于 b"
else
   echo "$a -eq $b: a 不等于 b"
fi
if [ $a -ne $b ]
then
   echo "$a -ne $b: a 不等于 b"
else
   echo "$a -ne $b : a 等于 b"
fi
if [ $a -gt $b ]
then
   echo "$a -gt $b: a 大于 b"
else
   echo "$a -gt $b: a 不大于 b"
fi
if [ $a -lt $b ]
then
   echo "$a -lt $b: a 小于 b"
else
   echo "$a -lt $b: a 不小于 b"
fi
if [ $a -ge $b ]
then
   echo "$a -ge $b: a 大于或等于 b"
else
   echo "$a -ge $b: a 小于 b"
fi
if [ $a -le $b ]
then
   echo "$a -le $b: a 小于或等于 b"
else
   echo "$a -le $b: a 大于 b"
fi

執(zhí)行腳本,輸出結(jié)果如下所示:

10 -eq 20: a 不等于 b
10 -ne 20: a 不等于 b
10 -gt 20: a 不大于 b
10 -lt 20: a 小于 b
10 -ge 20: a 小于 b
10 -le 20: a 小于或等于 b

布爾運(yùn)算符

下面列出了常用的布爾運(yùn)算符,假定變量 a 為 10,變量 b 為 20:

運(yùn)算符    說(shuō)明    舉例
!    非運(yùn)算,表達(dá)式為 true 則返回 false,否則返回 true。    [ ! false ] 返回 true。
-o    或運(yùn)算,有一個(gè)表達(dá)式為 true 則返回 true。    [ $a -lt 20 -o $b -gt 100 ] 返回 true。
-a    與運(yùn)算,兩個(gè)表達(dá)式都為 true 才返回 true。    [ $a -lt 20 -a $b -gt 100 ] 返回 false。

實(shí)例

布爾運(yùn)算符實(shí)例如下:

#!/bin/bash

a=10
b=20

if [ $a != $b ]
then
   echo "$a != $b : a 不等于 b"
else
   echo "$a != $b: a 等于 b"
fi
if [ $a -lt 100 -a $b -gt 15 ]
then
   echo "$a -lt 100 -a $b -gt 15 : 返回 true"
else
   echo "$a -lt 100 -a $b -gt 15 : 返回 false"
fi
if [ $a -lt 100 -o $b -gt 100 ]
then
   echo "$a -lt 100 -o $b -gt 100 : 返回 true"
else
   echo "$a -lt 100 -o $b -gt 100 : 返回 false"
fi
if [ $a -lt 5 -o $b -gt 100 ]
then
   echo "$a -lt 100 -o $b -gt 100 : 返回 true"
else
   echo "$a -lt 100 -o $b -gt 100 : 返回 false"
fi

執(zhí)行腳本,輸出結(jié)果如下所示:

10 != 20 : a 不等于 b
10 -lt 100 -a 20 -gt 15 : 返回 true
10 -lt 100 -o 20 -gt 100 : 返回 true
10 -lt 100 -o 20 -gt 100 : 返回 false

邏輯運(yùn)算符

以下介紹 Shell 的邏輯運(yùn)算符,假定變量 a 為 10,變量 b 為 20:

運(yùn)算符    說(shuō)明    舉例
&&    邏輯的 AND    [[ $a -lt 100 && $b -gt 100 ]] 返回 false
||    邏輯的 OR    [[ $a -lt 100 || $b -gt 100 ]] 返回 true

實(shí)例

邏輯運(yùn)算符實(shí)例如下:

#!/bin/bash

a=10
b=20

if [[ $a -lt 100 && $b -gt 100 ]]
then
   echo "返回 true"
else
   echo "返回 false"
fi

if [[ $a -lt 100 || $b -gt 100 ]]
then
   echo "返回 true"
else
   echo "返回 false"
fi

執(zhí)行腳本,輸出結(jié)果如下所示:

返回 false
返回 true

字符串運(yùn)算符

下面列出了常用的字符串運(yùn)算符,假定變量 a 為 "abc",變量 b 為 "efg":

運(yùn)算符    說(shuō)明    舉例
=    檢測(cè)兩個(gè)字符串是否相等,相等返回 true。    [ $a = $b ] 返回 false。
!=    檢測(cè)兩個(gè)字符串是否相等,不相等返回 true。    [ $a != $b ] 返回 true。
-z    檢測(cè)字符串長(zhǎng)度是否為0,為0返回 true。    [ -z $a ] 返回 false。
-n    檢測(cè)字符串長(zhǎng)度是否為0,不為0返回 true。    [ -n $a ] 返回 true。
str    檢測(cè)字符串是否為空,不為空返回 true。    [ $a ] 返回 true。

實(shí)例

#!/bin/bash

a="abc"
b="efg"

if [ $a = $b ]
then
   echo "$a = $b : a 等于 b"
else
   echo "$a = $b: a 不等于 b"
fi
if [ $a != $b ]
then
   echo "$a != $b : a 不等于 b"
else
   echo "$a != $b: a 等于 b"
fi
if [ -z $a ]
then
   echo "-z $a : 字符串長(zhǎng)度為 0"
else
   echo "-z $a : 字符串長(zhǎng)度不為 0"
fi
if [ -n $a ]
then
   echo "-n $a : 字符串長(zhǎng)度不為 0"
else
   echo "-n $a : 字符串長(zhǎng)度為 0"
fi
if [ $a ]
then
   echo "$a : 字符串不為空"
else
   echo "$a : 字符串為空"
fi

執(zhí)行腳本,輸出結(jié)果如下所示:

abc = efg: a 不等于 b
abc != efg : a 不等于 b
-z abc : 字符串長(zhǎng)度不為 0
-n abc : 字符串長(zhǎng)度不為 0
abc : 字符串不為空

文件測(cè)試運(yùn)算符

文件測(cè)試運(yùn)算符用于檢測(cè) Unix 文件的各種屬性。

屬性檢測(cè)描述如下:

操作符    說(shuō)明    舉例
-b file    檢測(cè)文件是否是塊設(shè)備文件,如果是,則返回 true。    [ -b $file ] 返回 false。
-c file    檢測(cè)文件是否是字符設(shè)備文件,如果是,則返回 true。    [ -c $file ] 返回 false。
-d file    檢測(cè)文件是否是目錄,如果是,則返回 true。    [ -d $file ] 返回 false。
-f file    檢測(cè)文件是否是普通文件(既不是目錄,也不是設(shè)備文件),如果是,則返回 true。    [ -f $file ] 返回 true。
-g file    檢測(cè)文件是否設(shè)置了 SGID 位,如果是,則返回 true。    [ -g $file ] 返回 false。
-k file    檢測(cè)文件是否設(shè)置了粘著位(Sticky Bit),如果是,則返回 true。    [ -k $file ] 返回 false。
-p file    檢測(cè)文件是否是具名管道,如果是,則返回 true。    [ -p $file ] 返回 false。
-u file    檢測(cè)文件是否設(shè)置了 SUID 位,如果是,則返回 true。    [ -u $file ] 返回 false。
-r file    檢測(cè)文件是否可讀,如果是,則返回 true。    [ -r $file ] 返回 true。
-w file    檢測(cè)文件是否可寫(xiě),如果是,則返回 true。    [ -w $file ] 返回 true。
-x file    檢測(cè)文件是否可執(zhí)行,如果是,則返回 true。    [ -x $file ] 返回 true。
-s file    檢測(cè)文件是否為空(文件大小是否大于0),不為空返回 true。    [ -s $file ] 返回 true。
-e file    檢測(cè)文件(包括目錄)是否存在,如果是,則返回 true。    [ -e $file ] 返回 true。

實(shí)例

變量 file 表示文件"/var/www/runoob/test.sh",它的大小為100字節(jié),具有 rwx 權(quán)限。下面的代碼,將檢測(cè)該文件的各種屬性:

#!/bin/bash

file="/var/www/runoob/test.sh"
if [ -r $file ]
then
   echo "文件可讀"
else
   echo "文件不可讀"
fi
if [ -w $file ]
then
   echo "文件可寫(xiě)"
else
   echo "文件不可寫(xiě)"
fi
if [ -x $file ]
then
   echo "文件可執(zhí)行"
else
   echo "文件不可執(zhí)行"
fi
if [ -f $file ]
then
   echo "文件為普通文件"
else
   echo "文件為特殊文件"
fi
if [ -d $file ]
then
   echo "文件是個(gè)目錄"
else
   echo "文件不是個(gè)目錄"
fi
if [ -s $file ]
then
   echo "文件不為空"
else
   echo "文件為空"
fi
if [ -e $file ]
then
   echo "文件存在"
else
   echo "文件不存在"
fi

執(zhí)行腳本,輸出結(jié)果如下所示:

文件可讀
文件可寫(xiě)
文件可執(zhí)行
文件為普通文件
文件不是個(gè)目錄
文件不為空
文件存在
echo命令

Shell 的 echo 指令與 PHP 的 echo 指令類(lèi)似,都是用于字符串的輸出。

命令格式:

echo string

您可以使用echo實(shí)現(xiàn)更復(fù)雜的輸出格式控制。

1.顯示普通字符串:

echo "It is a test"

這里的雙引號(hào)完全可以省略,以下命令與上面實(shí)例效果一致:

echo It is a test

2.顯示轉(zhuǎn)義字符

echo ""It is a test""

結(jié)果將是:

"It is a test"

同樣,雙引號(hào)也可以省略

3.顯示變量

read 命令從標(biāo)準(zhǔn)輸入中讀取一行,并把輸入行的每個(gè)字段的值指定給 shell 變量:

#!/bin/sh
read name 
echo "$name It is a test"

以上代碼保存為 test.sh,name 接收標(biāo)準(zhǔn)輸入的變量,結(jié)果將是:

[root@www ~]# sh test.sh
OK                     #標(biāo)準(zhǔn)輸入
OK It is a test        #輸出

4.顯示換行

echo -e "OK! 
" # -e 開(kāi)啟轉(zhuǎn)義
echo "It it a test"

輸出結(jié)果:

OK!

It it a test

5.顯示不換行

#!/bin/sh
echo -e "OK! c" # -e 開(kāi)啟轉(zhuǎn)義 c 不換行
echo "It is a test"

輸出結(jié)果:

OK! It is a test

6.顯示結(jié)果定向至文件

echo "It is a test" > myfile

7.原樣輸出字符串,不進(jìn)行轉(zhuǎn)義或取變量(用單引號(hào))

echo "$name""

輸出結(jié)果:

$name"

8.顯示命令執(zhí)行結(jié)果

  echo `date`

結(jié)果將顯示當(dāng)前日期

Thu Jul 24 10:08:46 CST 2014
printf 命令

printf 命令模仿 C 程序庫(kù)(library)里的 printf() 程序。

標(biāo)準(zhǔn)所定義,因此使用printf的腳本比使用echo移植性好。

printf 使用引用文本或空格分隔的參數(shù),外面可以在printf中使用格式化字符串,還可以制定字符串的寬度、左右對(duì)齊方式等。

默認(rèn)printf不會(huì)像 echo 自動(dòng)添加換行符,我們可以手動(dòng)添加 n。

printf 命令的語(yǔ)法:

printf  format-string  [arguments...]

參數(shù)說(shuō)明:

format-string: 為格式控制字符串

arguments: 為參數(shù)列表。

實(shí)例1

$ echo "Hello, Shell"
Hello, Shell
$ printf "Hello, Shell
"
Hello, Shell
$

接下來(lái),我來(lái)用一個(gè)腳本來(lái)體現(xiàn)printf的強(qiáng)大功能:

#!/bin/bash
 
printf "%-10s %-8s %-4s
" 姓名 性別 體重kg  
printf "%-10s %-8s %-4.2f
" 郭靖 男 66.1234 
printf "%-10s %-8s %-4.2f
" 楊過(guò) 男 48.6543 
printf "%-10s %-8s %-4.2f
" 郭芙 女 47.9876 

執(zhí)行腳本,輸出結(jié)果如下所示:

姓名     性別   體重kg
郭靖     男      66.12
楊過(guò)     男      48.65
郭芙     女      47.99
%s %c %d %f都是格式替代符。
%-10s 指一個(gè)寬度為10個(gè)字符(-表示左對(duì)齊,沒(méi)有則表示右對(duì)齊),任何字符都會(huì)被顯示在10個(gè)字符寬的字符內(nèi),如果不足則自動(dòng)以空格填充,超過(guò)也會(huì)將內(nèi)容全部顯示出來(lái)。
%-4.2f 指格式化為小數(shù),其中.2指保留2位小數(shù)。

實(shí)例2

#!/bin/bash
 
# format-string為雙引號(hào)
printf "%d %s
" 1 "abc"

# 單引號(hào)與雙引號(hào)效果一樣 
printf "%d %s
" 1 "abc" 

# 沒(méi)有引號(hào)也可以輸出
printf %s abcdef

# 格式只指定了一個(gè)參數(shù),但多出的參數(shù)仍然會(huì)按照該格式輸出,format-string 被重用
printf %s abc def

printf "%s
" abc def

printf "%s %s %s
" a b c d e f g h i j

# 如果沒(méi)有 arguments,那么 %s 用NULL代替,%d 用 0 代替
printf "%s and %d 
" 

執(zhí)行腳本,輸出結(jié)果如下所示:

1 abc
1 abc
abcdefabcdefabc
def
a b c
d e f
g h i
j  
 and 0

printf 的轉(zhuǎn)義序列

序列    說(shuō)明
a    警告字符,通常為ASCII的BEL字符
    后退
c    抑制(不顯示)輸出結(jié)果中任何結(jié)尾的換行字符(只在%b格式指示符控制下的參數(shù)字符串中有效),而且,任何留在參數(shù)里的字符、任何接下來(lái)的參數(shù)以及任何留在格式字符串中的字符,都被忽略
f    換頁(yè)(formfeed)

    換行

    回車(chē)(Carriage return)
	    水平制表符
v    垂直制表符
    一個(gè)字面上的反斜杠字符
ffffd    表示1到3位數(shù)八進(jìn)制值的字符。僅在格式字符串中有效