Shell 腳本中經(jīng)典的13個(gè)面試題
經(jīng)典的Shell十三問
1. 為何叫做shell?
2. Shell prompt(PS1)與Carriage Return(CR)的關(guān)系?
3. 別人echo, 你也echo, 是問echo知多少?
4. ""(雙引號(hào))與(單引號(hào))有什么區(qū)別?
5. var=value? export前后差在哪?
6. exec跟source差在哪?
7. ( ) 與 { } 差在哪?
8. () 還有 ${} 差在哪? 9. 與* 區(qū)別在哪? 10. && 與 || 差在哪? 11. > 與 < 差在哪? 12. 你要if還是case呢? 13. for what? while與until差在哪?
1. 為何叫做shell?
操作系統(tǒng)(OS,Opertating System)的軟件來管控。linux嚴(yán)格來說只是一個(gè)操作系統(tǒng)(OS),我們稱之為內(nèi)核(kernel)。kernel,而是通過kernel的外殼程序,也就是所謂的shell,來與kernel溝通。shell是一個(gè)使用者與系統(tǒng)的交互界面(interface), 只能讓使用者通過命令行(command line)來使用系統(tǒng)來完成工作。因此 ,shell最簡(jiǎn)單的定義就是——命令解譯器(Command Interpreter):
將使用者的命令翻譯給核心處理; 同時(shí),將核心處理結(jié)果翻譯給使用者。
OS使用不同的kernel;同一個(gè)
kernel之上,也可以使用不同的shell常見的
shell有sh; bash; csh; ksh;等2. Shell prompt(PS1)與Carriage Return(CR)的關(guān)系?
cursor左邊部分,稱之為提示符prompt通常一般用戶使用
$,管理員用戶root使用#shell prompt:可以輸入命令了
鍵入命令后,直到讀進(jìn)CR(Carriage Return)字符為止Carriage Return:可以執(zhí)行命令了
3. 別人echo, 你也echo, 是問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差在哪?
當(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í)行腳本
./sh
# 當(dāng)前shell執(zhí)行
source sh
# 當(dāng)前shell執(zhí)行后退出
exec sh
7. ( ) 與 { } 差在哪?
( ) 將command group置于sub-shell執(zhí)行{ } 則是在同一個(gè)shell內(nèi)完成8. () 還有 ${} 差在哪?
$()與 ``(反引號(hào)) 都是用來做命令替換用的。${var}與$var都是用來做變量替換用的。
# 假設(shè)我們定義了一個(gè)變量為:
file=/dir1/dir2/dir3/my.file.txt
# 我們可以用 ${ } 分別替換獲得不同的值:
# shell字符串的非貪婪(最小匹配)左刪除
${file#*/} # 拿掉第一條 / 及其左邊的字符串:dir1/dir2/dir3/my.file.txt
${file#*.} #拿掉第一個(gè) . 及其左邊的字符串:file.txt
# shell字符串的貪婪(最大匹配)左刪除
${file##*/} # 拿掉最后一條 / 及其左邊的字符串:my.file.txt
${file##*.} # 拿掉最后一個(gè) . 及其左邊的字符串:txt
# shell字符串的非貪婪(最小匹配)右刪除:
${file%/*} # 拿掉最后條 / 及其右邊的字符串:/dir1/dir2/dir3
${file%.*} # 拿掉最后一個(gè) . 及其右邊的字符串:/dir1/dir2/dir3/my.file
# shell字符串的貪婪(最大匹配)右刪除:
${file%%/*} # 拿掉第一條 / 及其右邊的字符串:(空值)
${file%%.*} # 拿掉第一個(gè) . 及其右邊的字符串:/dir1/dir2/dir3/my
記憶的方法為:
# 是去掉左邊(在鍵盤上 # 在 $ 之左邊)
% 是去掉右邊(在鍵盤上 % 在 $ 之右邊)
單一符號(hào)是最小匹配﹔兩個(gè)符號(hào)是最大匹配。
# shell字符串取子串:
${file:0:5}:提取最左邊的 5 個(gè)字節(jié):/dir1
${file:5:5}:提取第 5 個(gè)字節(jié)右邊的連續(xù) 5 個(gè)字節(jié):/dir2
# shell字符串變量值的替換:
${file/dir/path}:將第一個(gè) dir 提換為 path:/path1/dir2/dir3/my.file.txt
${file//dir/path}:將全部 dir 提換為 path:/path1/path2/path3/my.file.txt
# ${}還可針對(duì)變量的不同狀態(tài)(沒設(shè)定、空值、非空值)進(jìn)行賦值:
${file-my.file.txt} :假如 $file 沒有設(shè)定,則使用 my.file.txt 作傳回值。(空值及非空值時(shí)不作處理)
${file:-my.file.txt} :假如 $file 沒有設(shè)定或?yàn)榭罩担瑒t使用 my.file.txt 作傳回值。 (非空值時(shí)不作處理)
${file+my.file.txt} :假如 $file 設(shè)為空值或非空值,均使用 my.file.txt 作傳回值。(沒設(shè)定時(shí)不作處理)
${file:+my.file.txt} :若 $file 為非空值,則使用 my.file.txt 作傳回值。 (沒設(shè)定及空值時(shí)不作處理)
${file=my.file.txt} :若 $file 沒設(shè)定,則使用 my.file.txt 作傳回值,同時(shí)將 $file 賦值為 my.file.txt 。 (空值及非空值時(shí)不作處理)
${file:=my.file.txt} :若 $file 沒設(shè)定或?yàn)榭罩担瑒t使用 my.file.txt 作傳回值,同時(shí)將 $file 賦值為 my.file.txt 。 (非空值時(shí)不作處理)
${file?my.file.txt} :若 $file 沒設(shè)定,則將 my.file.txt 輸出至 STDERR。 (空值及非空值時(shí)不作處理)
${file:?my.file.txt} :若 $file 沒設(shè)定或?yàn)榭罩担瑒t將 my.file.txt 輸出至 STDERR。 (非空值時(shí)不作處理)
tips:
以上的理解在于, 你一定要分清楚 unset 與 null 及 non-null 這三種賦值狀態(tài).
一般而言, : 與 null 有關(guān), 若不帶 : 的話, null 不受影響, 若帶 : 則連 null 也受影響.
# 計(jì)算shell字符串變量的長度:${#var}
${#var} 可計(jì)算出變量值的長度:
${#file} 可得到 27 ,因?yàn)?nbsp;/dir1/dir2/dir3/my.file.txt 剛好是 27 個(gè)字節(jié)...
# bash數(shù)組(array)的處理方法
數(shù)組:
A=(a b c d)
引用數(shù)組:
${A[@]}
${A[*]}
訪問數(shù)組成員
${A[0]}
計(jì)算數(shù)組長度
${#A[@]}
${#A[*]}
數(shù)組重新賦值
A[2]=xyz
# $(( ))是用來做整數(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. && 與 || 差在哪?
test命令有兩種形式
test expression [ expression ]
bash的test目前支持三種測(cè)試對(duì)象
string:字符串 integer:整數(shù) file:文件
當(dāng)expression為真是返回 0(true) ,否則返回 非0(false)
command1 && command2# command2只有在command1的RV為0(true)的條件下執(zhí)行。command1 || command2# command2只有在command1的RV為非0(false)的條件下執(zhí)行。
先替換變量再比較
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)
我們可用
> 來改變送出的數(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)資料。
# 也就是說,在上例中,> 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
評(píng)論
圖片
表情
