關(guān)于 Shell 腳本的經(jīng)典十三問(wèn)
來(lái)源:https://www.cnblogs.com/rustling/p/9833174.html
1、為何叫做 Shell?
我們知道計(jì)算機(jī)的運(yùn)作離不開(kāi)硬件,但卻無(wú)法直接操作硬件,硬件的驅(qū)動(dòng)只能通過(guò)一種稱(chēng)之為“操作系統(tǒng)(OS,Opertating System)的軟件來(lái)管控。Linux 嚴(yán)格來(lái)說(shuō)是一個(gè)操作系統(tǒng)(OS)。
使用者沒(méi)有辦法直接操作 Kernel,而是通過(guò) Kernel 的“外殼”程序,也就是所謂的 Shell 來(lái)與 Kernel 溝通。Shell 是一個(gè)使用者與系統(tǒng)的交互界面(Interface),只能通過(guò)命令行(Command line)來(lái)使用系統(tǒng)來(lái)完成工作。因此 Shell 最簡(jiǎn)單的定義就是:命令解譯器(Command Interpreter)
將使用者的命令翻譯給內(nèi)核處理;
同時(shí),將內(nèi)核處理結(jié)果翻譯給使用者。
不同的 OS 使用不同的 Kernel;同一個(gè) kernel 之上,也可以使用不同的 Shell,常見(jiàn)的 Shell 有sh、bash、csh、ksh 等
2、Shell prompt(PS1)與 Carriage Return(CR)的關(guān)系?
成功登錄一個(gè) Shell 終端后,光標(biāo)左邊部分稱(chēng)之為提示符 Prompt,通常一般用戶(hù)使用$,管理員用戶(hù)使用#
Shell Prompt:可以輸入命令了,鍵入命令后,直到讀進(jìn) CR(Carriage Return)字符為止
Carriage Return:可以執(zhí)行命令了
若從技術(shù)的細(xì)節(jié)來(lái)看,Shell 會(huì)依據(jù) IFS(Internal Field Seperator)將 Command line 所輸入的文字拆解為”字段”(word/field)。然后再針對(duì)特殊字符(meta)先作處理,最后重組整行command line。
3、別人echo,你也echo,試問(wèn) echo 知多少?
echo 將 argument送到標(biāo)準(zhǔn)輸出(stdout),通常顯示在屏幕
stdin 標(biāo)準(zhǔn)輸入
stdout 標(biāo)準(zhǔn)輸出
stderr 標(biāo)準(zhǔn)錯(cuò)誤輸出
echo?-n??#?取消換行符
echo?-e??#?啟用反斜杠轉(zhuǎn)譯
4、雙引號(hào)""與單引號(hào)''有什么區(qū)別?
hard quote:
''(單引號(hào)),關(guān)閉所有引用soft quote:
""(雙引號(hào)),保留$引用
5、var=value? export 前后差在哪?
變量定義:
name=value,等號(hào)左右兩邊不能使用分隔符。變量替換:
echo ${name}export變量:
export name=value,使變量成為環(huán)境變量
#?本地變量
A=B
#?取消變量
unset?A
#?環(huán)境變量export?A=B
6、exec 跟 source 差在哪?
環(huán)境變量只能從父進(jìn)程到子進(jìn)程單向傳遞。換句話(huà)說(shuō):在子進(jìn)程中環(huán)境如何變更,均不會(huì)影響父進(jìn)程的環(huán)境。
當(dāng)我們執(zhí)行一個(gè)shell script時(shí),其實(shí)是先產(chǎn)生一個(gè)sub-shell的子進(jìn)程, 然后sub-shell再去產(chǎn)生命令行的子進(jìn)程。
#?創(chuàng)建子shell執(zhí)行腳本
./1.sh
#?當(dāng)前shell執(zhí)行
source?1.sh
#?當(dāng)前shell執(zhí)行后退出
exec?1.sh
7、( ) 與 { } 差在哪?
( )將 command group 置于 sub-shell 執(zhí)行{ }則是在同一個(gè)shell內(nèi)完成
8、$(()) 與 $() 還有 ${} 差在哪?
#?假設(shè)我們定義了一個(gè)變量為:
file=/dir1/dir2/dir3/my.file.txt
#?我們可以用?${?}?分別替換獲得不同的值:
#?1.?shell字符串的非貪婪(最小匹配)左刪除
${file#*/}?#?拿掉第一條?/?及其左邊的字符串:dir1/dir2/dir3/my.file.txt
#?2.?shell字符串的貪婪(最大匹配)左刪除
${file##*/}?#?拿掉最后一條?/?及其左邊的字符串:my.file.txt
${file##*.}?#?拿掉最后一個(gè) . 及其左邊的字符串:txt
#?3. shell字符串的非貪婪(最小匹配)右刪除:
${file%/*}?#?拿掉最后條?/?及其右邊的字符串:/dir1/dir2/dir3
${file%.*}?#?拿掉最后一個(gè) . 及其右邊的字符串:/dir1/dir2/dir3/my.file
#?4. shell字符串的貪婪(最大匹配)右刪除:
${file%%/*}?#?拿掉第一條?/?及其右邊的字符串:(空值)
${file%%.*}?#?拿掉第一個(gè) . 及其右邊的字符串:/dir1/dir2/dir3/my
記憶的方法為:
#?是去掉左邊(在鍵盤(pán)上?#?在?$?之左邊)
%?是去掉右邊(在鍵盤(pán)上?%?在?$?之右邊)
單一符號(hào)是最小匹配﹔兩個(gè)符號(hào)是最大匹配。
# 5. shell字符串取子串:
${file:0:5}:提取最左邊的 5 個(gè)字節(jié):/dir1
${file:5:5}:提取第 5 個(gè)字節(jié)右邊的連續(xù) 5 個(gè)字節(jié):/dir2
# 6. shell字符串變量值的替換:
${file/dir/path}:將第一個(gè) dir 替換為 path:/path1/dir2/dir3/my.file.txt
${file//dir/path}:將全部 dir 替換為 path:/path1/path2/path3/my.file.txt
# 7. ${}還可針對(duì)變量的不同狀態(tài)(沒(méi)設(shè)定、空值、非空值)進(jìn)行賦值:
${file-my.file.txt}?:假如?$file?沒(méi)有設(shè)定,則使用 my.file.txt 作傳回值。(空值及非空值時(shí)不作處理)?
${file:-my.file.txt}?:假如?$file?沒(méi)有設(shè)定或?yàn)榭罩?,則使用 my.file.txt 作傳回值。(非空值時(shí)不作處理)
${file+my.file.txt}?:假如?$file?設(shè)為空值或非空值,均使用 my.file.txt 作傳回值。(沒(méi)設(shè)定時(shí)不作處理)
${file:+my.file.txt}?:若?$file?為非空值,則使用 my.file.txt 作傳回值。(沒(méi)設(shè)定及空值時(shí)不作處理)
${file=my.file.txt}?:若?$file?沒(méi)設(shè)定,則使用?my.file.txt?作傳回值,同時(shí)將?$file?賦值為 my.file.txt 。(空值及非空值時(shí)不作處理)
${file:=my.file.txt}?:若?$file?沒(méi)設(shè)定或?yàn)榭罩担瑒t使用?my.file.txt?作傳回值,同時(shí)將?$file?賦值為 my.file.txt 。(非空值時(shí)不作處理)
${file?my.file.txt}?:若?$file?沒(méi)設(shè)定,則將 my.file.txt 輸出至 STDERR。(空值及非空值時(shí)不作處理)
${file:?my.file.txt}?:若?$file?沒(méi)設(shè)定或?yàn)榭罩?,則將 my.file.txt 輸出至 STDERR。(非空值時(shí)不作處理)
tips:
以上的理解在于,?你一定要分清楚?unset?與?null?及?non-null?這三種賦值狀態(tài).
一般而言,?:?與?null?有關(guān),?若不帶?:?的話(huà),?null?不受影響,?若帶?:?則連?null?也受影響.
#?8. 計(jì)算shell字符串變量的長(zhǎng)度:${#var}
${#var}?可計(jì)算出變量值的長(zhǎng)度:
${#file}?可得到?27?,因?yàn)?/dir1/dir2/dir3/my.file.txt?剛好是?27?個(gè)字節(jié)...
#?9.?bash數(shù)組(array)的處理方法
數(shù)組:
A=(a?b?c?d)
引用數(shù)組:
${A[@]}
${A[*]}
訪(fǎng)問(wèn)數(shù)組成員
${A[0]}
計(jì)算數(shù)組長(zhǎng)度
${#A[@]}
${#A[*]}
數(shù)組重新賦值
A[2]=xyz
#?10.$((?))是用來(lái)做整數(shù)運(yùn)算的?
a=5;b=7;c=2;
echo?$((?a?+?b?*?c))
9、$@ 與 $* 區(qū)別在哪?
"$@"則可得到 “p1” “p2 p3” “p4” 這三個(gè)不同的詞段"$*"則可得到 “p1 p2 p3 p4” 這一整串單一的詞段
10、&& 與 || 差在哪?
1、test命令有兩種形式
test expression
[ expression ]
2、bash的test目前支持三種測(cè)試對(duì)象
string:字符串
integer:整數(shù)
file:文件
3、當(dāng) expression 為真是返回 0(true) ,否則返回 非0(false)
command1 && command2 ?command2 只有在 command1 的RV為0(True)的條件下執(zhí)行。
command1 || command2 ?command2只有在command1的RV為非0(False)的條件下執(zhí)行。
4、先替換變量再比較
A=123[?-n?"$A"?]?&&?([?"$A"?-lt?100?]?||?echo?"too?big")unset?A
11、> 與 < 差在哪?
0: Standard Input(STDIN)
1: Standard Output (STDOUT)
2: Standard Error Output(STDERR)
我們可用 < 來(lái)改變讀進(jìn)的數(shù)據(jù)信道(stdin),使之從指定的檔案讀進(jìn)。
我們可用 > 來(lái)改變送出的數(shù)據(jù)信道(stdout, stderr),使之輸出到指定的檔案。
ls?my.file?no.such.file?1>?file.out?2>file.err
#?2>&1?就是將stderr并進(jìn)stdout做輸出
ls?my.file?no.such.file?1>?file.out?2>&1
#?/dev/null?空
ls?my.file?no.such.file?>/dev/null?2>&1
cat?file?>?file
#?在 IO Redirection 中,stdout 與 stderr 的管道會(huì)先準(zhǔn)備好,才會(huì)從 stdin 讀進(jìn)資料。
#?也就是說(shuō),在上例中,>?file?會(huì)先將?file?清空,然后才讀進(jìn)?file?,?
#?但這時(shí)候檔案已經(jīng)被清空了,因此就變成讀不進(jìn)任何數(shù)據(jù)了
12、你要if還是case呢?
#?if
echo?-n?"Do?you?want?to?continue?(Yes/No):"
read?YN
if?[?"$YN"=Y?-o?"$YN"=y?-o?"$YN"="Yes"?-o?"$YN"="yes"?-o?"$YN"="YES"];then
echo?"continue"
else
exit?0
fi
#?case
echo?-n?"Do?you?want?to?continue?(Yes/No):"
read?YN
case?"$YN"?in
[Yy]|[Yy][Ee][Ss])
echo?"continue"
;;
*)
exit?0
esac
13、for what? while與until差在哪?
#?for
for?((i=1;i<=10;i++))
do
echo?"num?is?$i"
done
#?while
num=1
while?[?"$num"?-le?10?];?do
echo?"num?is?$num"
num=$(($num?+?1))
done
#?until
num=1
until?[?"$num"?-gt?10?];?do
echo?"num?is?$num"
num=$(($nu?+?1))
done
break 是結(jié)束 loop
return 是結(jié)束 function
exit 是結(jié)束 script/shell
來(lái)源:https://www.cnblogs.com/rustling/p/9833174.html
有收獲,點(diǎn)個(gè)在看?




