一、變量的使用
1. 變量命名
定義變量時,變量名不加美元符号($,PHP語言中變量需要),如:
your_name="yikoulinux"
注意,變量名和等号之間不能有空格,這可能和你熟悉的所有程式設計語言都不一樣。同時,變量名的命名須遵循如下規則:
- 命名隻能使用英文字母,數字和下劃線,首個字元不能以數字開頭。
- 中間不能有空格,可以使用下劃線(_)。
- 不能使用标點符号。
- 不能使用bash裡的關鍵字(可用help指令檢視保留關鍵字)。
- 變量名稱一般習慣為大寫
有效的 Shell 變量名示例如下:
RUNOOB
LD_LIBRARY_PATH
_var
var2
無效的變量命名:
?var=123
user*name=runoob
2. 常用變量
Linux Shell 中的變量分為:系統變量和使用者自定義變量。
- 系統變量: H O M E 、 HOME、 HOME、PWD、 S H E L L 、 SHELL、 SHELL、USER 等等比如: echo $HOME 等等…
- 使用者自定義變量:
-
定義變量:變量=值
2)顯示目前 shell 中所有變量:set
3)撤銷變量:unset 變量
- 聲明靜态變量:readonly 變量,注意:不能 unset
- 将指令的傳回值賦給變量(重點)
除了顯式地直接指派,還可以用語句給變量指派,如:
1)
A=
ls -la
反引号,運作裡面的指令,并把結果傳回給變量 A
2)
A=$(ls -la)
$等價于反引号
3)
for file in `ls /etc`
或
for file in $(ls /etc)
以上語句将 /etc 下目錄的檔案名循環出來。
3. 舉例
例1:
含義如下:
- 定義一個變量名為name的變量,值為一口linux
- 輸出變量name的值
- 定義一個變量名為number的變量,初始值為22
- 輸出變量number的值
- 直接輸出帶變量的字元串
- 使用雙引号輸出帶變量的字元串
- 使用單引号輸出帶變量的字元串
- 使用雙引号輸出帶不存在的變量的字元串,不存在的變量預設為空
- 使用雙引号來聲明字元串中的變量
- 使用大括号{&變量名},聲明字元串中的變量
注意:
上述變量是臨時變量,當關閉終端後,變量就會消失。
例2:
删除變量并檢視指定變量
- unset name 删除變量name
- 檢視name變量
二、字元串的操作
在做shell批處理程式時候,經常會涉及到字元串相關操作。有很多指令語句,如:awk,sed都可以做字元串各種操作。
其實shell内置一系列操作符号,可以達到類似效果,大家知道,使用内部操作符會省略啟動外部程式等時間,是以速度會非常的快。
1. 字元串操作(長度,讀取,替換)
表達式 | 含義 |
---|---|
${#string} | $string的長度 |
${string:position} | 在 $string中, 從位置$position開始提取子串 |
${string:position:length} | 在$string中, 從位置$position開始提取長度為$length的子串 |
${string#substring} | 從變量$string的開頭, 删除最短比對$substring的子串 |
${string##substring} | 從變量$string的開頭, 删除最長比對$substring的子串 |
${string%substring} | 從變量$string的結尾, 删除最短比對$substring的子串 |
${string%%substring} | 從變量$string的結尾, 删除最長比對$substring的子串 |
${string/substring/replacement} | 使用$replacement, 來代替第一個比對的$substring |
${string//substring/replacement} | 使用$replacement, 代替所有比對的$substring |
${string/#substring/replacement} | 如果$string的字首比對$substring, 那麼就用$replacement來代替比對到的$substring |
${string/%substring/replacement} | 如果$string的字尾比對$substring, 那麼就用$replacement來代替比對到的$substring |
說明:"* $substring”可以是一個正規表達式.
2. 字元串操作舉例
a) 計算字元串長度
root@ubuntu:/home/peng# test='I love china'
root@ubuntu:/home/peng# echo ${#test}
12
${#變量名}得到字元串長度
b) 截取字串
root@ubuntu:/home/peng# test='I love china'
root@ubuntu:/home/peng# echo ${test:5}
e china
root@ubuntu:/home/peng# echo ${test:5:10}
e china
root@ubuntu:/home/peng#
root@ubuntu:/home/peng# echo ${test:4:10}
ve china
${變量名:起始:長度}得到子字元串
c) 字元串删除
root@ubuntu:/home/peng# test='c:/windows/boot.ini'
root@ubuntu:/home/peng# echo ${test#/}
c:/windows/boot.ini
root@ubuntu:/home/peng# echo ${test#*/}
windows/boot.ini
root@ubuntu:/home/peng# echo ${test##*/}
boot.ini
root@ubuntu:/home/peng# echo ${test%/*}
c:/windows
root@ubuntu:/home/peng# echo ${test%%/*}
c:
${變量名#substring正規表達式}從字元串開頭開始配備substring,删除比對上的表達式。
${變量名%substring正規表達式}從字元串結尾開始配備substring,删除比對上的表達式。
${test##*/},${test%/*} 分别是得到檔案名,或者目錄位址最簡單方法。
d) 字元串替換
root@ubuntu:/home/peng# test='c:/windows/boot.ini'
root@ubuntu:/home/peng# echo ${test/\//\\}
c:\windows/boot.ini
root@ubuntu:/home/peng# echo ${test//\//\\}
c:\windows\boot.ini
${變量/查找/替換值} 一個“/”表示替換第一個,”//”表示替換所有,當查找中出現了:”/”請加轉義符”\/”表示。
字元串的位置是從0開始,-1表示該字元串最後一個位置;
截取字元串的時候,是左閉右開的,從左邊的位置開始,一直到右邊的位置結束,不包括右邊的位置。
三、 腳本的建立和執行
shell腳本并不能作為正式的程式設計語言,因為它是在Linux的shell中運作的,是以稱他為shell腳本。
事實上,shell腳本就是一些指令的集合。
我們通常把所有的操作都記錄到一個文檔中,然後去調用文檔中的指令,這樣一步操作就可以完成了
一般shell腳本都是放在/usr/local/sbin的目錄下。
1) shell腳本的建立
在linux系統中,shell腳本(bash shell程式)通常是在編輯器(如vi/vim)中編寫,由unix/linux指令、bash shell指令、程式結構控制語句和注釋等内容組成,推薦用vim編輯器。
2) 腳本開頭(第一行)
一個規範的shell腳本的第一行會指出由哪個程式(解釋器)來執行腳本中的内容,在linux bash程式設計中一般為:
#!/bin/bash
#!/bin/sh <==255個字元以内
其中開頭的"#!“又稱為幻數,在執行bash腳本的時候,核心會根據”#!"後的解釋器來确定該用哪個程式解釋腳本中的内容,
這一行必須在每個腳本頂端的第一行,如果不是第一行則為腳本注釋行,例如下面的例子。
root@ubuntu:/home/peng# cat test1.sh
#!/bin/bash
echo "scajy start"
#!/bin/bash <==寫到這裡就是注釋
#!/bin/sh
echo "scajy en:"
sh和bash的差別
root@ubuntu:/home/peng# ls -l /bin/sh
lrwxrwxrwx 1 root root 4 Sep 21 2015 /bin/sh -> bash
提示:sh為bash的軟連接配接,這裡推薦用标準寫法#!/bin/bash
Bash是GNU/Linux預設的shell,和Bourne shell (sh)相容,Bash采用了Korn shell (Ksh)和C shell(csh)的特色。符合IEEE POISIX P10003.2/ISO 9945.2 shell and tools 标準。
Centos和redhat linux 下預設的shell 均為bash 是以,在寫shell腳本的時候,我們的腳本的開頭也可以不加#!/bin/bash。但如果目前的shell非你預設的shell時,比如tcsh,那麼久必須要寫#!了。否則腳本檔案就隻能執行一些指令的集合,不能夠使用shell内建的指令了,建議讀者養成習慣,不管什麼腳本最好都加上開頭語言辨別,這在後文的shell程式設計規範中會再次提到。
如果腳本的開頭不指定解析器,那麼,就要用對應的解釋器來執行腳本。例如:bash test.sh
3) 腳本注釋
在shell腳本中,跟在(#)#号後面的内容表示注釋,用來對腳本進行注釋說明,注釋部分不會被執行,僅僅是給人看的,注釋可自一行,也可以跟在腳本指令後面與指令在同一行,開發腳本時,如果沒有注釋,其他人就很難了解腳本究竟在做什麼,時間長了自己也會忘記。是以,我們要盡量成為所做的工作(腳本等)書寫注釋的習慣,不光是友善别人,也是友善自己。否則寫完一個腳本後也許後就記不起腳本的用途了,在重新閱讀也會浪費很多寶貴時間。對于團隊的協作也不利。
4) 舉例
建立檔案first.sh,并拷貝如下資訊到檔案:
#cd usr/local/sbin
# vim first.sh
#! /bin/bash
##this is my first shell script
#wirten by 一口Linux 2021.5.3
date
echo "Hello world"
shell腳本通常以.sh為字尾名
執行腳本
以下幾種方法都可以:
#sh first.sh
#bash first.sh
#./first.sh
#./first.sh
會報權限不夠
可以:
#chmod +x first.sh
四、環境變量的使用
1. 知識點詳解
1) 定義變量:變量=值
2)顯示目前 shell 中所有變量:set
3)撤銷變量:unset 變量
4) 聲明靜态變量:readonly 變量,注意:不能 unset
- 定義變量的規則
1) 變量名稱可以由字母、數字和下劃線組成,但是不能以數字開頭。
2) 等号兩側不能有空格
3) 變量名稱一般習慣為大寫
1)A=`ls -la` 反引号,運作裡面的指令,并把結果傳回給變量 A
2)A=$(ls -la) 等價于反引号
- 設定環境變量的基本文法:
export 變量名=變量值 (功能描述:将 shell 變量輸出為環境變量)
source 配置檔案(功能描述:讓修改後的配置資訊立即生效)
echo $變量名(功能描述:查詢環境變量的值)
2. 操作詳解
檢視環境變量HOME、PATH的值:
root@ubuntu:/home/peng# echo $HOME
/root
root@ubuntu:/home/peng# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/home/peng/toolchain/gcc-4.6.4/bin:/home/peng/toolchain/arm-cortex_a8/bin
檢視windows系統中的環境變量
檢視環境變量PATH中所有的路徑
腳本路徑安裝舉例
方法1:
修改環境環境變量:在PATH中添加指定“軟體安裝”的目錄:
root@ubuntu:/home/peng/yikou# pwd
/home/peng/yikou
root@ubuntu:/home/peng/yikou# ls
a.sh
root@ubuntu:/home/peng/yikou# sh a.sh
Sun May 2 17:00:14 PDT 2021
Hello world
root@ubuntu:/home/peng/yikou# export PATH=$PATH:/home/peng/yikou/
root@ubuntu:/home/peng/yikou# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/home/peng/toolchain/gcc-4.6.4/bin:/home/peng/toolchain/arm-cortex_a8/bin:/home/peng/yikou/
root@ubuntu:/home/peng/yikou# a.sh
bash: /home/peng/yikou/a.sh: 權限不夠
root@ubuntu:/home/peng/yikou# chmod 777 a.sh
root@ubuntu:/home/peng/yikou# a.sh
Sun May 2 17:01:34 PDT 2021
Hello world
方法2:
修改環境變量配置檔案的方式,使修改後的環境變量永久生效
vim /etc/bash.bashrc
source .bash.rc ,使配置檔案重新生效
關閉終端,打開并重新重新輸入:a.sh依舊可以執行。
root@ubuntu:/home/peng/# a.sh
Sun May 2 17:10:00 PDT 2021
Hello world
五、數學運算
運算符使用的文法:
expr操作符對照表
操作符含義
- 比較大小,隻能對整數進行比較,需要加空格,linux 保留關鍵字要轉義
root@ubuntu:/home/peng/yikou# num1=30
root@ubuntu:/home/peng/yikou# num2=50
root@ubuntu:/home/peng/yikou# expr $num1 \> $num2
0
檢視上一條指令有沒有執行成功:
傳回0 成功,其他失敗
- 小于、小于等于、大于等于
expr $num1 \< $num2
expr $num1 \<= $num2
expr $num1 \>= $num2
- 運算 加、減、乘、除
# 加
num1=17
num2=5
expr $num1 + $num2
# 減
num3=`expr $num1 + $num2`
echo $num3
expr $num1 - $num2
# 乘
expr $num1 \* $num2
expr $num1 / $num2
# 取餘數
expr $num1 % $num2
需要注意事項:
兩個小括号的計算方法,要指派,否則會報錯
六、腳本與使用者互動
操作指令行參數
1. 讀取參數
bash shell用位置參數變量(positional parameter)存儲指令行輸入的所有參數,包括程式名。
其中,$0表示程式名,$1表示第1個參數,$2表示第2個參數,…, 9 表 示 第 9 個 參 數 。 如 果 參 數 個 數 多 于 9 個 , 必 須 如 下 表 示 變 量 : 9表示第9個參數。如果參數個數多于9個,必須如下表示變量: 9表示第9個參數。如果參數個數多于9個,必須如下表示變量:{10},${11},…
#!/bin/bash
# author:一口Linux
for((count = 1; count <= $1; count++))
do
echo The number is $count.
done
執行結果:
修改腳本如下:
echo $1, $2, $3, $4
執行結果如下:
2 讀取程式名
首先想到的是利用$0,但是$0擷取的檔案名包括./以及路徑等字首資訊,如下:
echo The command entered is: $0
# 運作:./
# 輸出:The command entered is: ./14.sh
如果想僅得到檔案名,而不包含./,可以使用basename指令:
name=`basename $0`
echo The command entered is: $name
# 運作:./
# 輸出:The command entered is: 14.sh
3 特殊變量
$#表示指令行參數的個數:
#!/bin/bash
# author:一口Linux
params=$#
echo The number of params is: $params
for((i = 1; i <= params; i++))
do
echo The param is: $i
done
執行結果
如果想擷取所有的參數,當然可以利用KaTeX parse error: Expected 'EOF', got '#' at position 1: #̲和循環逐個周遊。也可以利用如下…*将所有的指令行參數看作一個整體存儲,而$@将指令行中以空格間隔的參數單獨存儲,如下:
#!/bin/bash
# author:一口Linux
count=1
for param in "$*"
do
echo "\$* parameter $count = $param"
count=$[ $count + 1 ]
done
count=1
for param in "$@"
do
echo "\$@ parameter $count = $param"
count=$[ $count + 1 ]
done
4 基本的讀取
read指令接受從鍵盤或檔案描述符中的輸入資料,将其存儲到一個指定變量中。
選項:
-p:指定讀取值時的提示符;
-t:指定讀取值時等待的時間(秒),如果沒有在指定的時間内輸入,就不再等待了。。
-n:設定允許輸入字元的個數
參數
變量:指定讀取值的變量名
操作詳解
- 例1
#!/bin/bash
# author:一口Linux
# testing the read option
read -p "Please enter your name: " name
echo "Hello $name."
-
例2
read指令中,可以根據需要将輸入的資料儲存在多個變量中,如果指定的變量比較少,那麼最後一個變量将包含餘下的所有輸入,如下所示:
#!/bin/bash
# author:一口Linux
# testing the read option
read -p "Enter the values: " val1 val2 val3
echo "$val1"
echo "$val2"
echo "$val3"
-
綜合例子
提示使用者輸入一個正整數num,然後計算1+2+3+…+num的值;必須對num是否為正整數做判斷,不符合應當運作再次輸入
思路:
- expr隻能對整數進行計算,直接用expr 和一個整數計算擷取 $? 的值來判斷是否為整數
- 在使用 expr $num1 > 0 判斷是否大于0
#!/bin/bash
# author:一口Linux
while true
do
read -p "please input a positive number: " num
# 判斷數是否是整數
expr $num + 1 &> /dev/null
if [ $? -eq 0 ];then
# 判斷這個整數是否大于0,大于0傳回1
if [ `expr $num \> 0` -eq 1 ];then
#echo "yes,positive number"
# $sum沒有指派,預設為0
for i in `seq 0 $num`
do
sum=`expr $sum + $i`
done
echo "1+2+3+...+$num = $sum"
# 執行計算需要退出
exit
fi
fi
echo "error,input enlegal"
continue
done
測試:
七、關系運算符
有時候我們需要比較兩個數字的大小關系,這時候就要用到關系運算符。關系運算符隻支援數值運算,不支援字元運算。
Shell 語言支援下面這些關系運算符:
-eq:檢測兩個數是否相等,相等傳回 true。
-ne:檢測兩個數是否不相等,相等傳回 true。
-gt:檢測左邊的數是否大于右邊的,如果是傳回 true。
-lt:檢測左邊的數是否小于右邊的,如果是傳回 true。
-ge:檢測左邊的數是否大于等于右邊的,如果是傳回 true。
-le:檢測左邊的數是否小于等于右邊的,如果是傳回 true。
#!/bin/bash
# author:一口Linux
a=10
b=20
if [ $a -gt $b ]
then
echo "a great than b"
else
echo "a not great than b"
fi
執行結果!如下:
八、字元串運算符
= 比較兩個字元串是否相等
!= 比較兩個字元串是否不相等
-z 檢測字元串的長度是否為零
-n 檢測字元串的長度是否不為零
$字元名 變量是否有負值(為空),有傳回True,沒有傳回False
#!/bin/bash
# author:一口Linux
###本腳本主要用于字元串運算符
if [ ! $1 ]
then
echo "第一個參數為空"
echo "****************************************************************"
echo "****************************************************************"
echo "**************執行用例的格式為:sh $0 變量1 變量2***************"
echo "****************************************************************"
echo "****************************************************************"
break
else
if [ ! $2 ]
then
echo "第二個參數為空"
echo "****************************************************************"
echo "****************************************************************"
echo "**************執行用例的格式為:sh $0 變量1 變量2***************"
echo "****************************************************************"
echo "****************************************************************"
break
else
###1、檢測兩個字元串是否相等;
if [ $1 = $2 ]
then
echo "這是第一個判斷語句"
echo "變量1等于變量2"
else
echo "這是第一個判斷語句"
echo "變量1不等于變量2"
fi
###2、檢測兩個字元串是否不相等;
if [ $1 != $2 ]
then
echo "這是第二個判斷語句"
echo "變量1不等于變量2"
else
echo "這是第二個判斷語句"
echo "便量1等于變量2"
fi
###3、檢測字元串長度是否為0
if [ -z $1 ]
then
echo "這是第三個判斷段語句"
echo "變量1字元串長度為0"
else
echo "這是第三個判斷段語句"
echo $1
fi
###4、檢測字元串長度是否不為0
if [ -n $2 ]
then
echo "這是第四個判斷語句"
echo "變量2字元串長度不為0"
echo $2
else
echo "這是第四個判斷語句"
echo "變量2字元串長度為0"
fi
###5、檢測字元串是否不為空
if [ $1 ]
then
echo "這是第五個判斷語句"
echo "變量1不為空"
else
echo "這是第五個判斷語句"
echo "變量1為空"
fi
fi
fi
九、shell檔案及目錄常用操作的幾個執行個體
-
提取路徑的目錄和檔案名
提取目錄:
dirname $path
basename $path
- 批量重命名帶有空格檔案
function processFilePathWithSpace(){
find $1 -name "* *" | while read line
do
newFile=`echo $line | sed 's/[ ][ ]*/_/g'`
mv "$line" $newFile
logInfo "mv $line $newFile $?"
done
}
- 周遊檔案内容
cat /tmp/text.txt | while read line
do
echo $line
done
- 檔案不存在,則建立檔案
[ -f $logFile ] || touch $logFile
- 遞歸周遊目錄
function getFile(){
for file in `ls $1`
do
element=$1"/"$file
if [ -d $element ]
then
getFile $element
else
echo $element
fi
done
}
- 清空檔案内容
cat /dev/null > $filePath