Shell 指令碼中經典的13個面試題

語言: CN / TW / HK
  • 經典的Shell十三問

    • 1. 為何叫做shell?

    • 2. Shell prompt(PS1)與Carriage Return(CR)的關係?

    • 3. 別人echo, 你也echo, 是問echo知多少?

    • 4. ""(雙引號)與(單引號)有什麼區別?

    • 5. var=value? export前後差在哪?

    • 6. exec跟source差在哪?

    • 7. ( ) 與 { } 差在哪?

    • 8. 與() 還有 ${} 差在哪?

    • 9. 與* 區別在哪?

    • 10. && 與 || 差在哪?

    • 11. > 與 < 差在哪?

    • 12. 你要if還是case呢?

    • 13. for what? while與until差在哪?

1. 為何叫做shell?

我們知道計算機的運作不能離開硬體,但使用者卻無法直接操作硬體,硬體的驅動只能通過一種稱為 作業系統 ( OSOpertating System )的軟體來管控。 linux 嚴格來說只是一個 作業系統 ( OS ),我們稱之為 核心 ( kernel )。

使用者沒有辦法直接操作一個 kernel ,而是通過 kernel 的外殼程式,也就是所謂的 shell ,來與 kernel 溝通。 shell 是一個使用者與系統的 互動介面(interface) , 只能讓使用者通過 命令列 ( command line )來使用系統來完成工作。因此 ,
shell最簡單的定義就是——命令解譯器(Command Interpreter):

  • 將使用者的命令翻譯給核心處理;

  • 同時,將核心處理結果翻譯給使用者。

不同的 OS 使用不同的 kernel ;
同一個 kernel 之上,也可以使用不同的 shell
常見的 shellsh; bash; csh; ksh;

2. Shell prompt(PS1)與Carriage Return(CR)的關係?

成功登入一個shell終端後,遊標 cursor 左邊部分,稱之為提示符 prompt
通常一般使用者使用 $ ,管理員使用者 root 使用 #

  • shell prompt:可以輸入命令了
    鍵入命令後,直到讀進
    CR(Carriage Return) 字元為止

  • Carriage Return:可以執行命令了

若從技術的細節來看,shell會依據IFS(Internal Field Seperator) 將command line所輸入的文字拆解為"欄位"(word/field)。然後再針對特殊字元(meta)先作處理,最後重組整行command line。

3. 別人echo, 你也echo, 是問echo知多少?

echoargument 送到 標準輸出(stdout) ,通常顯示在螢幕

  • stdin 標準輸入

  • stdout 標準輸出

  • stderr 標準錯誤輸出

echo -n  # 取消換行符echo -e  # 啟用反斜槓轉譯

4. ""(雙引號)與(單引號)有什麼區別?

  • hard quote: '' (單引號),關閉所有引用

  • soft quote: "" (雙引號),保留 $ 引用

5. var=value? export前後差在哪?

變數定義: name=value ,等號左右兩邊不能使用分隔符。
變數替換: echo ${name}
export變數: export name=value ,使變數成為環境變數

# 本地變數A=B# 取消變數unset A# 環境變數export A=B

6. exec跟source差在哪?

環境變數只能從父程序到子程序單向傳遞。換句話說:在子程序中環境如何變更,均不會影響父程序的環境。
當我們執行一個shell script時,其實是先產生一個sub-shell的子程序, 然後sub-shell再去產生命令列的子程序。

# 建立子shell執行指令碼./sh# 當前shell執行source sh# 當前shell執行後退出exec sh

7. ( ) 與 { } 差在哪?

( )command group 置於 sub-shell 執行
{ } 則是在同一個 shell 內完成

8.與() 還有 ${} 差在哪?

  • $() 與 ``(反引號) 都是用來做命令替換用的。

  • ${var} 與  $var 都是用來做變數替換用的。

# 假設我們定義了一個變數為:file=/dir1/dir2/dir3/my.file.txt# 我們可以用 ${ } 分別替換獲得不同的值:#  shell字串的非貪婪(最小匹配)左刪除${file#*/} # 拿掉第一條 / 及其左邊的字串:dir1/dir2/dir3/my.file.txt${file#*.} #拿掉第一個 . 及其左邊的字串:file.txt#  shell字串的貪婪(最大匹配)左刪除${file##*/} # 拿掉最後一條 / 及其左邊的字串:my.file.txt${file##*.} # 拿掉最後一個 . 及其左邊的字串:txt#  shell字串的非貪婪(最小匹配)右刪除:${file%/*} # 拿掉最後條 / 及其右邊的字串:/dir1/dir2/dir3${file%.*} # 拿掉最後一個 . 及其右邊的字串:/dir1/dir2/dir3/my.file#  shell字串的貪婪(最大匹配)右刪除:${file%%/*} # 拿掉第一條 / 及其右邊的字串:(空值)${file%%.*} # 拿掉第一個 . 及其右邊的字串:/dir1/dir2/dir3/my記憶的方法為:# 是去掉左邊(在鍵盤上 # 在 $ 之左邊)% 是去掉右邊(在鍵盤上 % 在 $ 之右邊)單一符號是最小匹配﹔兩個符號是最大匹配。
#  shell字串取子串:${file:0:5}:提取最左邊的 5 個位元組:/dir1${file:5:5}:提取第 5 個位元組右邊的連續 5 個位元組:/dir2#  shell字串變數值的替換:${file/dir/path}:將第一個 dir 提換為 path:/path1/dir2/dir3/my.file.txt${file//dir/path}:將全部 dir 提換為 path:/path1/path2/path3/my.file.txt#  ${}還可針對變數的不同狀態(沒設定、空值、非空值)進行賦值:${file-my.file.txt} :假如 $file 沒有設定,則使用 my.file.txt 作傳回值。(空值及非空值時不作處理) 
${file:-my.file.txt} :假如 $file 沒有設定或為空值,則使用 my.file.txt 作傳回值。(非空值時不作處理)${file+my.file.txt} :假如 $file 設為空值或非空值,均使用 my.file.txt 作傳回值。(沒設定時不作處理)${file:+my.file.txt} :若 $file 為非空值,則使用 my.file.txt 作傳回值。(沒設定及空值時不作處理)${file=my.file.txt} :若 $file 沒設定,則使用 my.file.txt 作傳回值,同時將 $file 賦值為 my.file.txt 。(空值及非空值時不作處理)${file:=my.file.txt} :若 $file 沒設定或為空值,則使用 my.file.txt 作傳回值,同時將 $file 賦值為 my.file.txt 。(非空值時不作處理)${file?my.file.txt} :若 $file 沒設定,則將 my.file.txt 輸出至 STDERR。(空值及非空值時不作處理)${file:?my.file.txt} :若 $file 沒設定或為空值,則將 my.file.txt 輸出至 STDERR。(非空值時不作處理)

tips:
以上的理解在於, 你一定要分清楚 unset 與 null 及 non-null 這三種賦值狀態.
一般而言, : 與 null 有關, 若不帶 : 的話, null 不受影響, 若帶 : 則連 null 也受影響.
#  計算shell字串變數的長度:${#var}${#var} 可計算出變數值的長度:${#file} 可得到 27 ,因為 /dir1/dir2/dir3/my.file.txt 剛好是 27 個位元組...#  bash陣列(array)的處理方法陣列:
A=(a b c d)
引用陣列:
${A[@]}
${A[*]}
訪問陣列成員
${A[0]}
計算陣列長度
${#A[@]}${#A[*]}陣列重新賦值
A[2]=xyz# $(( ))是用來做整數運算的 a=5;b=7;c=2;
echo $(( a + b * c))

9.與* 區別在哪?

  • "$@" 則可得到 "p1" "p2 p3" "p4" 這三個不同的詞段

  • "$*" 則可得到 "p1 p2 p3 p4" 這一整串單一的詞段

10. && 與 || 差在哪?

  1. test命令有兩種形式

  • test expression

  • [ expression ]

  1. bash的test目前支援三種測試物件

  • string:字串

  • integer:整數

  • file:檔案

  1. 當expression為真是返回 0(true) ,否則返回 非0(false)

  • command1 && command2 # command2只有在command1的RV為0(true)的條件下執行。

  • command1 || command2 # command2只有在command1的RV為非0(false)的條件下執行。

  1. 先替換變數再比較

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)

我們可用 < 來改變讀進的資料通道(stdin),使之從指定的檔案讀進。
我們可用
> 來改變送出的資料通道(stdout, stderr),使之輸出到指定的檔案。

ls my.file no.such.file 1> file.out 2>file.err# 2>&1 就是將stderr並進stdout做輸出ls my.file no.such.file 1> file.out 2>&1# /dev/null 空ls my.file no.such.file >/dev/null 2>&1cat < file > file# 在 IO Redirection 中,stdout 與 stderr 的管道會先準備好,才會從 stdin 讀進資料。# 也就是說,在上例中,> file 會先將 file 清空,然後才讀進 < file , # 但這時候檔案已經被清空了,因此就變成讀不進任何資料了

12. 你要if還是case呢?

# ifecho -n "Do you want to continue?(Yes/No):"read YNif [ "$YN"=Y -o "$YN"=y -o "$YN"="Yes" -o "$YN"="yes" -o "$YN"="YES"];thenecho "continue"elseexit 0fi# caseecho -n "Do you want to continue?(Yes/No):"read YNcase "$YN" in[Yy]|[Yy][Ee][Ss])echo "continue";;
*)exit 0esac

13. for what? while與until差在哪?

# forfor ((i=1;i<=10;i++))do
 echo "num is $i"done# whilenum=1while [ "$num" -le 10 ]; do
 echo "num is $num"
 num=$(($num + 1))done# untilnum=1
until [ "$num" -gt 10 ]; do
 echo "num is $num"
 num=$(($nu + 1))done

break 是結束 loop

return 是結束 function

exit 是結束 script/shell

end

一口Linux 

關注,回覆【 1024 】海量Linux資料贈送

精彩文章合集

文章推薦

【專輯】 ARM

【專輯】 粉絲問答

【專輯】 所有原創

專輯 linux 入門

專輯 計算機網路

專輯 Linux驅動

【乾貨】 嵌入式驅動工程師學習路線

【乾貨】 Linux嵌入式所有知識點-思維導圖

【課程推薦】

針對應屆生和轉行的朋友,彭老師錄製了 基於Linux的物聯網綜合實戰課程 。以專案為基礎講解知識點,掌握就有2年以上工作經驗。

課程詳情 《一個適合寫到簡歷的基於linux的物聯網綜合專案

報名聯絡: yikoupeng(微訊號)

課程 報名 及開發板 購買 ,後臺回覆關鍵字: 無線感測網

點選“ 閱讀原文 ”檢視更多分享,歡迎 點分享、收藏、點贊、在看