shell程式設計基礎
- 變量:字元串,數字,環境和參數
- 條件:shell中的布爾值
- 程式控制:if,elif,for,while,until,case
- 指令清單
- 函數
- shell内置指令
- 擷取指令的執行結果
shell腳本是一個包含一系列指令的文本檔案.
文法:
#!/bin/sh
符号#!用來指定該腳本的解析程式.
修改腳本的屬性:
chmod +x filename
變量
1.引号
#!/bin/sh
myvar="Hi there"
echo $myvar
echo "$myvar"
echo '$myvar'
echo \$myvar
echo Enter some text
read myvar
echo '$myvar' now equals $myvar
exit 0
![](https://img.laitimes.com/img/9ZDMuAjOiMmIsIjOiQnIsICOxczNwYTNwEjMwUDM1EDMy8CX0Vmbu4GZzNmLn9Gbi1yZtl2Lc9CX6MHc0RHaiojIsJye.jpg)
分析: $獲得變量的内容,使用雙引号,不影響變量的替換,使用單引号,反斜線不進行變量的替換.
2.參數變量
參數變量 | 說明 |
---|---|
$# | 傳入腳本的指令行參數個數 |
$* | 是以指令行參數,各個值之間有空格 |
[email protected] | 各個參數擴充彼此分開的域,不受IFS值的影響 |
$0 | 指令本身(shell檔案名) |
$1 | 第一個指令參數 |
$2 | 第二個指令行參數 |
案例:
#!/bin/sh
echo "number of vars:"$#
echo "values of vars:"$*
echo "value of var1:"$1
echo "value of var2:"$2
echo "value of var3:"$3
echo "value of var4:"$4
條件
(1)
if [ expression ]
then
#code block
fi
(2)
if [ expression ]
then
# code block
else
# code block
fi
(3)
if [ expression ]
then
# code block
else if [ pression ]
then
# code block
else
# code block
fi
fi
(4)
if [ expression ];then
# code block
elif [ expression ]
then
# code block
else
# code blcok
fi
fi
字元串比較
字元串比較 | 結果 |
---|---|
string1 = string2 | 兩個字元串相同,結果為真 |
string1 != string2 | 兩個字元串不同,結果為真 |
-n string | 字元串不為空,結果為真 |
-z string | 如果字元串為null,則結果為真 |
算術比較
算術比較 | 結果 |
---|---|
ex1 -eq ex2 | 相同結果為真 |
ex1 -ne ex2 | 不相等結果為真 |
ex1 -gt ex2 | ex1 > ex2為真 |
ex1 -ge ex2 | ex1 >= ex2 為真 |
ex1 -lt ex2 | ex1 < ex2 為真 |
ex1 -le ex2 | ex1 <= ex2 為真 |
!ex | 表達式結果為假,結果為真 |
綜合比較
比較操作 | 整數比較 | 字元串比較 |
---|---|---|
相同 | -eq | = |
不同 | -ne | != |
大于 | -gt | > |
小于 | -lt | < |
大于或等于 | -ge | |
小于或等于 | -le | |
為空 | -z | |
不為空 | -n |
檔案條件測試
檔案條件測試 | 結果 |
---|---|
-d file | 檔案為目錄為真 |
-e file | 檔案存在 |
-f file | 檔案時普通檔案 |
-g file | 檔案的set-group-id位被設定為真 |
-r file | 檔案可讀 |
-s file | 檔案大小不為0 |
-u file | 檔案的set-urer-id位被設定為真 |
-w file | 檔案可寫 |
-x file | 檔案可執行 |
案例代碼:
#!/bin/sh
if [ -f /bin/bash ]
then
echo "file /bin/bash exists"
fi
if [ -d /bin/bash ]
then
echo "/bin/bash is a directory"
else
echo "/bin/bash is NOT a directory "
fi
控制結構
1.for循環
for var in [list] do
done- code block
其中$var是循環控制變量,[list]是var需要周遊的一個集合,do/done對包含了循環體,相當于C語言中一對大括号.若do 和 for 寫在同一行上,需要用”;”.
案例代碼:
#! /bin/sh
for day in Sun Mon Tue Wed Thu Fri Sat
do
echo $day
done
exit 0
注意: 在for循環那行,變量day是沒有加” "符号的,而在循環體内,echo所在行變量 day是必須加上”$”符号的.
代碼案例:列印目前目錄下f開頭的所有腳本檔案
#!/bin/sh
for file in $(ls f*.sh)
do
echo $file
done
exit 0
2.while循環
while [ condition ]
do
# code block
done
#! /bin/sh
echo "Enter password"
read trythis
while [ "$trythis" != "secret" ];do
echo "Sorry,try again"
read trythis
done
exit 0
3.until循環
until [ condition ]
do
# code block
done
差別: while 和 until 的差別在于while 為真時執行,until為假時執行.
4. case 語句
case “$var” in
condition1 )
;;
condition2 )
;;
* )
default statments
;;
esac
#! /bin/sh
echo "Hit a key,then hit return. "
read Keypress
case "$Keypress" in
[A-Z]) echo "Uppercase letter"
;;
[a-z]) echo "Lowercase letter"
;;
[0-9]) echo "Digit"
;;
*) echo"Punctuation,whitespace,or other"
;;
esac
exit 0
函數
函數的定義方式
function_name(){
statements
}
#! /bin/sh
yes_or_no()
{
echo "Is your name $* ?"
while true
do
echo -n "Enter yes or no:"
read x
case "$x" in
y | yes ) return 0 ;;
n | no ) return 1 ;;
* ) echo "Answer yes or no" ;;
esac
done
}
echo "Original parameters are $*"
if yes_or_no "$1"
then
echo "Hi $1,nice name"
else
echo "Never mind"
fi
exit 0
指令
1. break指令
預設情況下,break隻跳出一層循環.
2. : 指令
冒号(:)指令隻是一個空指令.
相當于true的别名. while : 實作無限循環
: 結構也會用在變量的條件設定中.
: $ {var:=value}
3. continue指令
使得for,while,untill循環跳到下一次循環繼續執行.
4. .指令
點(.)指令用于目前shell中的執行指令.
5.echo指令
使用echo指令輸出結尾帶有換行符的的字元串.
去掉換行符:
(1)使用 -n
echo -n "string to output"
(2)使用 -e 啟動反斜線轉義字元(\c代表去掉換行符,\t代表制表符,\n代表回車)
echo -e "string to output\c"
6.eval指令
eval指令允許對參數進行求值.
foo=10
x=foo
y='$'$x
echo $y # 輸出為 $foo
foo=10
x=foo
eval y='$'$x
echo $y # 輸出為10
7.exec指令
兩種不同的用法
(1)将目前shell替換成一個不同的程式.
(2)修改目前檔案描述符
8.exit指令
exit指令使得腳本以退出碼n結束運作.
在退出碼0代表成功.
退出碼1~125是腳本程式使用的錯誤代碼.
退出碼 | 說明 |
---|---|
126 | 檔案不可執行 |
127 | 指令未找到 |
128以上 | 出現一個信号 |
9.export指令
export:将自定義變量設定為環境變量,這個環境變量可以被目前調用的其他腳本和程式看到.(目前shell中有效)
10.expr指令
expr指令将它的參數當做一個表達式求值.
expr 2 + 3
11.printf指令
格式化輸出,和C語言中的有點像
$ printf "%s\n" hello
hello
$ printf "%s %d\t%s" "Hi There" 15 people
Hi There 15 people
字元轉換限定符 | 說明 |
---|---|
d | 十進制輸出 |
c | 輸出一個字元 |
s | 輸出一個字元串 |
12.return指令
return指令的作用使函數傳回值.
13.set指令
set指令的作用是為shell設定參數變量.
案例代碼:
#! /bin/sh
echo the date is $(date)
set $(date)
echo $0
echo The year is: $1
echo The month is: $2
echo The day is: $3
exit 2
14.shift指令
把所有參數變量左移一個位置,使得$2變成$1,$3變成$2
15.trap指令
trap指令的一種常見用途是在腳本程式被中斷時完成清理工作.
trap command signal
兩個參數,一個是接收到指定信号采取的行動,一個是處理的信号名.
信号 | 說明 |
---|---|
HUP(1) | 挂起 |
INT(2) | 中斷,Ctrl+C |
QUIT(3) | 退出,Ctrl+\ |
ABRT(6) | 終止,嚴重執行錯誤發送 |
ALRM(14) | 報警,處理逾時 |
TERM(15) | 終止,系統關機發送 |
#! /bin/sh
trap 'rm -f /tmp/my_tmp_file_$$' INT # 捕捉中斷信号,捕捉到後執行指令,但是程式可以繼續運作.
echo creating file /tmp/my_tmp_file_$$
date > /tmp/my_tmp_file_$$
echo "press interrupt (CTRL-C) to interrupt...."
while [ -f /tmp/my_tmp_file_$$ ];do
echo File exists
sleep 1
done
echo The file no longer exists
trap INT #隻捕捉到INT信号,但是不執行任何指令.采取預設處理方式,立即終止腳本
echo creating file /tmp/my_tmp_file_$$
date > /tmp/my_tmp_file_$$
echo "presss interrrupt (CTRL-C) to interrrupt..."
while [ -f /tmp/my_tmp_file_$$ ];do
echo File exists
sleep 1
done
echo we never get here
exit 0
16.unset指令
unset指令的作用是從環境中删除變量和函數.
#! /bin/sh
foo="Hello world"
echo $foo
unset foo
echo $foo
說明:和foo= 執行結果一樣,但是并不等同.foo=把foo設定設定為空.但是foo變量仍然存在,使用unset foo 效果是把變量foo從環境中删除.
參考書籍:
Linux 程式設計