-
經(jīng)典的Shell十三問(wèn)
-
1. 為何叫做shell?
-
2. Shell prompt(PS1)與Carriage Return(CR)的關(guān)系?
-
3. 別人echo, 你也echo, 是問(wèn)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?
我們知道計(jì)算機(jī)的運(yùn)作不能離開(kāi)硬件,但使用者卻無(wú)法直接操作硬件,硬件的驅(qū)動(dòng)只能通過(guò)一種稱為
操作系統(tǒng)
(OS
,Opertating System
)的軟件來(lái)管控。linux
嚴(yán)格來(lái)說(shuō)只是一個(gè)操作系統(tǒng)
(OS
),我們稱之為內(nèi)核
(kernel
)。使用者沒(méi)有辦法直接操作一個(gè)
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):-
將使用者的命令翻譯給核心處理; -
同時(shí),將核心處理結(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)
cursor
左邊部分,稱之為提示符prompt
通常一般用戶使用$
,管理員用戶root
使用#
-
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)譯
登錄后復(fù)制
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
登錄后復(fù)制
6. exec跟source差在哪?
環(huán)境變量只能從父進(jìn)程到子進(jìn)程單向傳遞。換句話說(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)程。另外,搜索公眾號(hào)Linux就該這樣學(xué)后臺(tái)回復(fù)“git書籍”,獲取一份驚喜禮包。# 創(chuàng)建子shell執(zhí)行腳本 ./sh # 當(dāng)前shell執(zhí)行 source sh # 當(dāng)前shell執(zhí)行后退出 exec sh
登錄后復(fù)制
7. ( ) 與 { } 差在哪?
( )
將command group
置于sub-shell
執(zhí)行{ }
則是在同一個(gè)shell
內(nèi)完成8. 與() 還有 ${} 差在哪?
-
$()
與 “(反引號(hào)) 都是用來(lái)做命令替換用的。 -
${var}
與$var
都是用來(lái)做變量替換用的。# 假設(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)是最大匹配。
登錄后復(fù)制
# 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)(沒(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)榭罩担瑒t使用 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)榭罩担瑒t將 my.file.txt 輸出至 STDERR。 (非空值時(shí)不作處理) tips: 以上的理解在于, 你一定要分清楚 unset 與 null 及 non-null 這三種賦值狀態(tài). 一般而言, : 與 null 有關(guān), 若不帶 : 的話, null 不受影響, 若帶 : 則連 null 也受影響.
登錄后復(fù)制
# 計(jì)算shell字符串變量的長(zhǎng)度:${#var} ${#var} 可計(jì)算出變量值的長(zhǎng)度: ${#file} 可得到 27 ,因?yàn)?/dir1/dir2/dir3/my.file.txt 剛好是 27 個(gè)字節(jié)... # bash數(shù)組(array)的處理方法 數(shù)組: A=(a b c d) 引用數(shù)組: ${A[@]} ${A[*]} 訪問(wèn)數(shù)組成員 ${A[0]} 計(jì)算數(shù)組長(zhǎng)度 ${#A[@]} ${#A[*]} 數(shù)組重新賦值 A[2]=xyz # $(( ))是用來(lái)做整數(shù)運(yùn)算的 a=5;b=7;c=2; echo $(( a + b * c))
登錄后復(fù)制
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í)行。關(guān)注Linux中文社區(qū) -
command1 || command2
# command2只有在command1的RV為非0(false)的條件下執(zhí)行。-
先替換變量再比較
A=123 [ -n "$A" ] && ([ "$A" -lt 100 ] || echo "too big") unset A
登錄后復(fù)制
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ù)了
登錄后復(fù)制
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
登錄后復(fù)制
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
登錄后復(fù)制
-
break 是結(jié)束 loop -
return 是結(jié)束 function -
exit 是結(jié)束 script/shell 以上就是你應(yīng)該知道的Shell 腳本的經(jīng)典十三問(wèn)的詳細(xì)內(nèi)容,更多請(qǐng)關(guān)注www.92cms.cn其它相關(guān)文章!
-
-
-
-
-
-
-
-
-
-
-
-