Linux命令行與Shell腳本編程大全讀書筆記

Linux內核4大主要功能:

內存管理

進程管理

設備管理

文件系統管理


Linux系統啟動的進程和腳本管理

1./etc/inittab ? 管理系統開機時會自動啟動的進程

2./etc/init.d ? ? 管理開機時啟動或停止某個應用的腳本放在這個目錄下,/etc/rcX.d目錄在啟動時,所有加載的腳本都是init.d下腳本的link


Linux 命令集合

1.ipcs -m? 查看系統上的當前共享內存頁面


2.cp -R dir1 dir2:? 遞歸的復制dir1dir2

?? cp -f test* dir2:將所有以test開頭的文件復制到dir2-f參數用來強制覆蓋dir2目錄中已有的同名文件,而不會提示用戶

?? cp -a old/. new/:將old目錄下的所有文件拷貝到new,包括隱藏文件

?? cp -l test1 test4: 創建了test1的硬鏈接文件test4,實際上二者標示的是同一個文件,因二者的節點號是相同的

?? cp -s test1 test5:創建了test1的軟鏈接文件test5。鏈接文件只要存儲源文件的信息,并不需要存儲源文件中的數據


?? ln -s test1 test5? ? ? ln -l test1 test4? 可以達到和上述類似的效果


  1. rm -rf dir2:刪除dir2目錄及其子目錄和文件
  2. stat test1:查看文件統計信息
  3. file test1:查看文件類型,包括ASCII文本文件;Shell腳本文件;二進制可執行文件
  4. cat -n 1.txt? 查看1.txt的內容,且給文本的每行加上行號?

?? ? ? cat -b 1.txt? 與上述指令的區別,只在文本非空白行加上行號

?? ? ? cat -s 1.txt? 壓縮文本中連續的多個空白行為一個空白行

?? ? ? more 1.tx? 查看文本文件的內容,空格鍵可以翻頁; 回車鍵可以顯示下一行

  1. ps -aux:可以查看當前系統所有進程的詳細信息

?? ? ? ps l: 查看當前控制臺下屬于當前用戶的進程信息

?? ? ? 可以在上述指令后增加參數 - -forest,可以看到進程之間的啟動關系

?? ? ? 進程信息中STAT列含義如下:

?? ? ? <:該進程運行在高優先級

?? ? ? N:該進程運行在低優先級

?? ? ? L:該進程有頁面鎖定在內存中

?? ? ? s:該進程是個控制進程

?? ? ? l:該進程是多線程的

?? ? ? +:該進程運行在前端

  1. top指令設置刷新間隔為2: top -d 2?
  2. df -h: 顯示某個設備上還有多少磁盤空間
  3. du -h: 顯示某一個目錄下各個子目錄的空間
  4. printenv:打印當前shell所在的全局變量
  5. set:顯示為某個特定進程設置的所有環境變量(包括全局變量)
  6. export 變量名:可以導出當前的局部變量成為全局變量 ?如:export test
  7. 變量定義注意事項:環境變量名稱、等號和值之間沒有空格;如果賦值一個含有空格的字符串,必須用單引號將字符串擴起來

?15. unset:刪除環境變量,如:unset test.?注意:在shell子進程中unset的全局環境變量,在父進程中該環境變量依然有效。

  1. PATH:定義了命令行輸入命令的搜索路徑。
  2. /etc/profile 是每個shell啟動時,默認的主啟動文件
  3. 每個用戶的shell啟動腳本,有如下三者中的一種:$HOME/.bash_profile ? $HOME/.bash_login ? $HOME/.profile?這三個文件會加載$HOME目錄下的.bashrc的啟動腳本,.bashrc會加載/etc目錄下的bashrc文件,同時$/HOME/.bashrc可以作為用戶自定義腳本執行的地方。
  4. alias -p 可以查看已有的命令別名列表
  5. 查看Flash分區信息:cat /proc/mtd(顯示分區名稱、大小)? ? cat /proc/partition(顯示block)
  6. shell中顯示提示而不換行,如:echo -n “This is a test string: ”
  7. shell中顯示$符號,需要按照轉義處理。如:echo “test print \$15”
  8. shell中使用反引號,可以獲取命令的輸出結果,然后賦值給一個變量。如:test=`date` ? test變量內容即為date的輸出
  9. IFS:內部字段分隔符,該環境變量定義了bash shell用作字段分隔符的一系列字符。可以如下方式使用該變量:

IFS.OLD=$IFS

IFS=$’\n’

<……use the new IFS value in code>

IFS=$IFS.OLD

  1. $*變量:把所有參數當成一個參數,而不是多個參數; ? $@變量:會單獨處理每個參數,所有參數當做同一字符串中的多個獨立的單詞
  2. 最后一個參數的表現方式:

params=$#

echo the last parameter is $params

echo the last parameter is ${!#}

  1. 使用$和方括號將數學表達式圈起來:var1=$[1+(2*5)]
  2. bc:浮點計算器
  3. $?: 該變量用來保存上個執行命令的退出狀態碼,退出碼0,表示命令成功結束;0,表示命令異常結束
  4. exit:該命令可以在shell結束時,指定一個退出狀態碼。該值取值范圍0~255,如果指定返回大于255,會被系統自動取模后的結果,作為退出狀態碼
  5. if-then語句格式一:

if command

then

? ? ? commands

fi


  1. if-then語句格式二(command之后增加一個分號)

if command: then

? ? ? commands

fi


  1. if-then-else格式:

if command

then

? ? commands

else

? ? commands

fi


  1. 嵌套if( elif )

if command1

then

? ? commands

elif command2

then

? ? commands

fi


  1. test:上述if的條件判斷都是根據命令的執行退出狀態碼來執行,如果想執行條件判斷,需要使用test命令,或者使用[ ]
  2. if條件判斷命令格式一:

if test condition

then

? ? commands

fi


  1. if條件判斷命令格式二:

if [ condition ]

then

? ? commands

fi


  1. test命令可以判斷3類條件:數值比較、字符串比較、文件比較
  2. test數值比較功能:

n1 -eq n2? ? ? ? ? ? ? ? ? 檢查n1是否等于n2

n1 -ge n2? ? ? ? ? ? ? ? ? 檢查n1是否大于或等于n2

n1 -gt? n2? ? ? ? ? ? ? ? ? 檢查n1是否大于n2

n1 -le n2 ? ? ? ? ? ? ? ? ? 檢查n1是否小于或等于n2

n1 -lt n2? ? ? ? ? ? ? ? ? ? 檢查n1是否小于n2

n1 -ne n2? ? ? ? ? ? ? ? ? 檢查n1是否不等于n2


  1. test字符串比較功能:

str1 = str2 ? ? ? ? ? ? ? ? 檢查str1是否與str2相同

str1 != str2? ? ? ? ? ? ? ? 檢查str1是否與str2不同

str1 < str2 ? ? ? ? ? ? ? ? 檢查str1是否比str2小(在使用時,需要對于符號轉義,如:if [ $val1 \< $val2 ]

str1 > str2 ? ? ? ? ? ? ? ? 檢查str1是否比str2大(在使用時,需要對于符號轉義,如:if [ $val1 \> $val2 ]

-n str1 ? ? ? ? ? ? ? ? ? ? ? 檢查str1的長度是否大于0

-z str1 ? ? ? ? ? ? ? ? ? ? ? 檢查str1的長度是否為0

注:在test命令中,大寫字母會小于小寫字母,這是因為test使用標準ASCII順序,即小寫字母的ASCII值會大于大寫字母的ASCII

sort命令排序時,小寫字母的會排在前面


  1. test文件較功能:

-d file? ? ? ? ? ? ? ? ? ? ? ? 檢查file是否存在并且是一個目錄

-e file? ? ? ? ? ? ? ? ? ? ? ? 檢查file是否存在

-f? file? ? ? ? ? ? ? ? ? ? ? ? 檢查file是否存在并且是一個文件

-s file? ? ? ? ? ? ? ? ? ? ? ? 檢查file是否存在并且非空

-r file ? ? ? ? ? ? ? ? ? ? ? ? 檢查file是否存在并且可讀

-w file ? ? ? ? ? ? ? ? ? ? ? 檢查file是否存在并且可寫

-x file? ? ? ? ? ? ? ? ? ? ? ? 檢查file是否存在并且可執行

-O file ? ? ? ? ? ? ? ? ? ? ? 檢查file是否存在并屬于當前用戶所有

-G file ? ? ? ? ? ? ? ? ? ? ? 檢查file是否存在并與當前用戶有相同的組

file1 -nt file2 ? ? ? ? ? ? 檢查file1是否比file2

file1 -ot file2 ? ? ? ? ? ? 檢查file1是否比file2


-b file ? ? 若文件存在且是一個塊特殊文件,則為真

-c file ? ? 若文件存在且是一個字符特殊文件,則為真

-d file ? ? 若文件存在且是一個目錄,則為真

-e file ? ? 若文件存在,則為真

-f file ? ? 若文件存在且為一個規則文件,則為真

-g file ? ? 若文件存在且設置了SGID位的值,則為真

-h file ? ? 若文件存在且為一個符號鏈接,則為真

-k file ? ? 若文件存在且設置了“sticky”位的值,則為真

-p file ? ? 若文件存在且為一已命名管道,則為真

-s file ? ? 若文件存在且其大小大于零,則為真

-u file ? ? 若文件存在且設置了SUID位的值,則為真

-r file ? ? 若文件存在且可讀,則為真

-w file ? ? 若文件存在且可寫,則為真

-x file ? ? 若文件存在且可執行,則為真

-o file ? ? 若文件存在且被有效用戶ID所擁有,則為真


$# 是傳給腳本的參數個數

$0 是腳本本身的名字

$1 是傳遞給該shell腳本的第一個參數

$2 是傳遞給該shell腳本的第二個參數? (注意:對于多過9個參數,需要在數字周圍加花括號,如第10個參數: ${10})

$@ 是傳給腳本的所有參數的列表

$* 是以一個單字符串顯示所有向腳本傳遞的參數,與位置變量不同,參數可超過9

$$ 是腳本運行的當前進程ID

$? 是顯示最后命令的退出狀態,0表示沒有錯誤,其他表示有錯誤

$-記錄著當前設置的shell選項


  1. 復合條件測試:

[ condition1 ] && [ condition2 ]? ? 只有在 && 左邊的命令返回真(命令返回值 $? == 0),&&右邊的命令才會被執行

[ condition1 ] || [ condition2 ]


  1. test使用標準數學運算符

if (( $val1 ** 2 > 90))? ? ? ? ? 雙括號中的特殊字符,如 < > 不再需要轉義,就可以直接使用了


  1. 雙括號中命令符號

val++? ? ? ? ? ? ? ? ? ? ? ? 后增

val- - ? ? ? ? ? ? ? ? ? ? ? ? 后減

++val? ? ? ? ? ? ? ? ? ? ? ? 先增

- -val ? ? ? ? ? ? ? ? ? ? ? ? 先減

** ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 冪運算

<< ? ? ? ? ? ? ? ? ? ? ? ? ? ? 左移位

>> ? ? ? ? ? ? ? ? ? ? ? ? ? ? 右移位

&& ? ? ? ? ? ? ? ? ? ? ? ? ? ? 邏輯與

||? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 邏輯或

? ? ? ? ? ? ? ? ? ? ? ? ? ? ?邏輯非

& ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 位與

|? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 位或

~ ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? 位求反


  1. test使用字符串模式匹配比較

if [[ $USER == r* ]]? ? ? ? $USER環境變量是否是以字母r開頭的


  1. case命令

case variable in

pattern1 | pattern2)

? ? commands1;;

pattern3)

? ? commands2;;

*)

? ? default commands;;

esac


  1. for命令

for var in list

do

? ? commands

done


  1. for命令中使用通配符

for file in /home/rich/test/*

do

? ? echo “test file $file”

done


  1. C語言風格for命令

for (( i=1; i <= 10; i++))

do

? ? echo “test…”

done


  1. while 命令

while test command

do

? ? other commands

done

如:

while [ $val1 -gt 0]

do

? ? echo $val1

? ? var1=$[$var1 - 1]

done

注意:while命令允許你在while語句行定義多個測試命令,但只有最后一個測試命令的退出狀態碼會被用來決定什么時候結束循環


51.until 命令

until test command

do

? ? other commands

done

注意:until命令要求你指定一個通常輸出非零退出碼的測試命令。(非零,表示測試條件尚不成立)


52.break 跳出多重循環

break n

這里n 說明了要跳出的循環層級。默認情況下,n1,表明跳出的是當前的循環。

如果將n 設置為2break命令就會停止下一級的外部循環


  1. continue 繼續循環

continue n

這里n 定義了要繼續的循環層級。

continue 2 表示跳出當前循環,繼續外層的循環


54.要在shell腳本的參數值中包含空格,需要把有空格的參數用單引號或雙引號括起來


55.basename命令會只返回程序名而不包括路徑

如: name=`basename $0`


56.shift 命令

默認情況下,這個命令會將命令前移,如,變量$3的值會移到$2

shift的好處:不知道參數個數,可以利用shift指令,只操作第一個參數$1,只需要在循環中每次shift,同時判斷$1是否為空(while [ -n “$1” ]

shift n

表明要移動幾個位置,如shift 2 : 移動2個位置


注意:當使用shift時,一個參數被移除后,它的值會被丟掉,且無法恢復



57.getopt? 命令

處理命令行選項和參數時非常方便的工具

格式:

getopt options? optstring parameters

options:如 -q選項,可以在命令執行時忽略錯誤信息

optstring是這個過程的關鍵,定義了命令行有效的選項字母,還定義了那些選項字母需要參數值。每個需要參數的選項字母后加一個冒號。

例如:

getopt -q ab:cd? -a -b test1 -c -d test2 test3


執行結果:-a -b test1 -c -d? - - test2 test3

這里- - 表示分開行中的額外參數


  1. getopts 命令

getopt的命令中,選項字母的參數值是不能有空格的,而 getopts就可以解決該問題

格式:

getopts opstring variable?

opstring: 有效字母都會列在其中,如果選項字母要求有個參數,就加一個冒號。要去掉錯誤消息,可以在opstring之前加一個冒號。

例如:

getopts :ab:c? -a -b “test1 test2“ -c


59.read命令

1)基本的讀取

例如1

echo -n “Enter you name: ”? ? -n 選項會移除掉字符串末尾的換行符,允許用戶緊跟其后輸入數據

read name ? ? ? ? ? ? ? ? ? ? ? ? ? ? 讀取的數據存儲到$name中了


例如2

read -p “please enter you age: ” age ? ? -p選項,允許你直接在read命令行中指定提示字符,讀取的數據存儲到 $age


(2)超時處理

例如1

read -t 5 -p “please enter you name: ” name? ? ? -t選項指定了read命令等待輸入的秒數


例如2

read -n1 -p “Do you want to continue [Y/N] ?” answer ? ? ? ? -n1選項,表明read讀取單個字符后退出


3)隱藏讀取(多用于密碼)

read -s -p “Enter you password: ” pass? ? ? ? -s選項,阻止將數據顯示在顯示器上,實際上只是將輸入文本顏色設置成和背景色一樣罷了


4)從文件中讀取

cat test | while read line? ? ? ? ? ? ? ? 將從cat test文件內容,每次read都讀取一行,將一行內容存儲到變量$line


60.查看系統的所有線程

1ps -T

2top之后輸入h,可以查看各個線程對于CPU的占用


61.重定向描述符

例如1:同時重定向錯誤和數據

ls -al test test2 test3 2>test6 1>test7

重定向錯誤到test6重定向輸出到test7


例如2:重定向輸入

cat < testfile

強制cat命令接收來自非STDIN的文件的輸入


62.重定向輸出和錯誤 ? &>

例如:

ls -al test test2 test3 &> test7

ls的結果,包括可能的錯誤輸出都重定向到文件test7


63.臨時重定向輸出到錯誤STDERR? ? >&2

例如:

echo “This is an error” >&2


64.永久重定向? exec命令

shell腳本中,使用exec命令,永久重定向shell執行期間的特定描述符

例如1:腳本中重定向輸出

exec 1>testout

exec命令會啟動一個新shell并將STDOUT文件描述符重定向到文件。腳本中所有發給STDOUT的輸出,都會被重定向到testout文件中


例如2:腳本中重定向輸入

exec 0< testfile

這個命令告訴shell它應該從文件testfile中獲得輸入,而不是STDIN


65.阻止命令輸入,重定向到/dev/null

這種情況,在腳本運行在后臺時很常見。shell輸入到null的任何數據都不會保存,這些數據都被丟掉了。

例如1

ls -al > /dev/null


例如2:阻止任何錯誤都不保存

ls -al badfile test6 2> /dev/null


例如3>/dev/null?2>&1?

shell中可能經常能看到:>/dev/null?2>&1?


命令的結果可以通過%>的形式來定義輸出


/dev/null?代表空設備文件

>?代表重定向到哪里,例如:echo?"123"?>?/home/123.txt

1?表示stdout標準輸出,系統默認值是1,所以">/dev/null"等同于"1>/dev/null"

2?表示stderr標準錯誤

&?表示等同于的意思,2>&1,表示2的輸出重定向等同于1


那么本文標題的語句:

1>/dev/null?首先表示標準輸出重定向到空設備文件,也就是不輸出任何信息到終端,說白了就是不顯示任何信息。

2>&1?接著,標準錯誤輸出重定向等同于?標準輸出,因為之前標準輸出已經重定向到了空設備文件,所以標準錯誤輸出也重定向到空設備文件。


66.快速移除現有文件數據

輸入重定向時,將/dev/null作為輸入文件。由于/dev/null文件不含有任何內容,程序員通常用它來快速移除現有文件中的數據,而不用先刪除再創建文件。

例如:

cat /dev/null > testfile

文件testfile仍然存在系統上,但現在它是空文件。


67.mktemp命令

/tmp目錄下創建臨時文件

例如1:創建本地臨時文件

mktemp testing.XXXXXX

mktemp命令會用6個字符碼替換這6X,從而保證文件名在目錄中是唯一的。


例如2:創建臨時文件時,返回全路徑

tempfile = `mktemp -t tmp.XXXXXX`

后續可以在任何目錄下,通過tempfile來引用新創建的臨時文件了,因此時獲知了全路徑


例如3:創建臨時目錄

tempdir =`mktemp -d dir.XXXXXX`

后續可以再tempdir這個臨時目錄下創建自己的臨時文件。


68.tee命令

將輸入一邊發送到顯示器,一邊發送到tee命令指定的文件中

例如1

date | tee testfile

date的執行結果,既顯示在屏幕上,又存儲到文件testfile


例如2tee追加存儲

date | tee -a testfile

testfile中追加新的date的數據。沒有-a選項,則每次執行tee,都會覆蓋文件的內容


69.信號處理

SIGHUP? ? ? ? 掛起進程,當終端斷開時,將發送該信號給終端控制進程

SIGINT? ? ? ? ? 終止進程(ctrl + c)終端驅動程序將發送該信號給前臺進程組

SIGQUIT ? ? ? 停止進程(ctrl + \該信號將發往前臺進程組,并生成可用于調試的核心轉儲文件。在進程陷入死循環時,可以觸發該信號調試

SIGKILL ? ? ? ? 無條件終止進程 ?

SIGTERM? ? ? 可能的話終止進程? killkillall命令所發送的默認信號。

SIGSTOP ? ? ? 無條件停止進程,但不是終止進程

SIGTSTP? ? ? ? 停止或暫停進程,但不終止進程(ctrl + z?作業控制的停止信號,發送信號給前臺進程組,使其停止運行。

SIGCONT ? ? ? 繼續運行停止的進程


shell默認情況下,會忽略收到的 SIGQUIT SIGTERM信號。正因為這樣,交互式shell才不會被意外終止。


用戶有時會使用kill -KILL 或者killall -9 顯式向進程發送SIGKILL信號。然而這一做法通常是錯誤的。精心的設計應用程序應當為SIGTERM信號設置處理器程序,以便于其能夠預先清除臨時文件和釋放其他資源,從而全身而退。

發送SIGKILL信號可以殺掉某個進程,從而繞開了SIGTERM信號的處理程序。因此,總是應該首先嘗試使用SIGTERM信號來終止進程,而把SIGKILL信號作為最后手段,去對付那些不響應SIGTERM信號的失控進程


SIGKILL SIGSTOP信號,是一個必執行信號。處理器程序無法將其阻塞、忽略或者捕獲,故而總能成功執行。


  1. shell中捕捉信號

trap commands signals

例如:

trap “echo ‘SorryI have trapped Ctrl-C’”? SIGINT? SIGTERM

trap在每次檢測到SIGINT SIGTERM信號時顯示一行簡單的文本消息。


一般trap 指令放置在shell執行的正文第一行,在捕捉到信號后,自動執行。


  1. grep命令

grep -q -e "No medium found" -e "No such file or directory"

說明:

-q:安靜模式,不打印任何標準輸出。如果有匹配的內容則立即返回狀態值0

-e:并列使用多個 -e參數可以實現或條件,同時查詢多個條件,滿足其中之一即可。


72.函數

(1)函數定義兩種形式:

function name {

commands

}


name( ){

commands

}


(2)函數的返回值

情況一:

函數中最后一條命令的退出狀態碼,作為函數的退出狀態碼


情況二:

return 命令來退出函數并返回特定的退出狀態碼

注:函數的退出狀態碼必須小于256,大于256將返回一個錯誤


(3)函數的輸出保存到shell變量中

如:result=`testfunc`

這條命令將testfunc函數的輸出付給$result變量。

這種方法,可以用來從函數返回大于256的返回值,只需要在函數的最后用echo將結果輸出即可。

如:echo $[$value * 2]


(4)函數中使用變量

情況一:

向函數傳遞變量,bash shell會將函數當做小腳本來對待。

如:

value=`addfunc 10 15`

向函數addfunc傳遞了兩個參數10和15


情況二:

由于函數使用特殊參數環境變量作為自己的參數值,它不能直接從腳本的命令行獲取腳本的參數值。即腳本的參數如$1,與函數的$1不同。


情況三:全局變量和局部變量

在shell腳本中任何地方定義的變量都是全局變量;

在函數內部使用local聲明的變量,都是局部變量。

例如:local temp


  1. shell中的${},##和%%的使用?

假設我們定義了一個變量為:
file=/dir1/dir2/dir3/my.file.txt

可以用${ }分別替換得到不同的值:
${file#*/}:刪掉第一個 /及其左邊的字符串:dir1/dir2/dir3/my.file.txt
${file##*/}:刪掉最后一個 /?及其左邊的字符串:my.file.txt
${file#*.}:刪掉第一個 .?及其左邊的字符串:file.txt
${file##*.}:刪掉最后一個 .?及其左邊的字符串:txt
${file%/*}:刪掉最后一個? /?及其右邊的字符串:/dir1/dir2/dir3
${file%%/*}:刪掉第一個 /?及其右邊的字符串:(空值)
${file%.*}:刪掉最后一個? .?及其右邊的字符串:/dir1/dir2/dir3/my.file
${file%%.*}:刪掉第一個? . ?及其右邊的字符串:/dir1/dir2/dir3/my

記憶的方法為:
# 去掉左邊(鍵盤上# $的左邊)
%是去掉右邊(鍵盤上% $ 的右邊)
單一符號是最小匹配;兩個符號是最大匹配
${file:0:5}:提取最左邊的 5個字節:/dir1
${file:5:5}:提取第 5個字節右邊的連續5個字節:/dir2

也可以對變量值里的字符串作替換:
${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 (非空值時不作處理)

${#var} 可計算出變量值的長度:

${#file} 可得到 27,因為/dir1/dir2/dir3/my.file.txt27個字節



Shell重定向&>file、2>&1、1>&2的區別


0表示標準輸入
1表示標準輸出
2表示標準錯誤輸出
> 默認為標準輸出重定向,與 1> 相同
2>&1 意思是把 標準錯誤輸出 重定向到 標準輸出.
&>file 意思是把 標準輸出 和 標準錯誤輸出 都重定向到文件file中




1>/dev/null 2>&1的含義

shell中可能經常能看到:>/dev/null 2>&1

命令的結果可以通過%>的形式來定義輸出

/dev/null
代表空設備文件
>
代表重定向到哪里,例如:echo "123" > /home/123.txt
1
表示stdout標準輸出,系統默認值是1,所以">/dev/null"等同于"1>/dev/null"
2
表示stderr標準錯誤
&
表示等同于的意思,2>&1,表示2的輸出重定向等同于1

那么本文標題的語句:
1>/dev/null
首先表示標準輸出重定向到空設備文件,也就是不輸出任何信息到終端,說白了就是不顯示任何信息。
2>&1
接著,標準錯誤輸出重定向等同于 標準輸出,因為之前標準輸出已經重定向到了空設備文件,所以標準錯誤輸出也重定向到空設備文件。

?

/dev/null 的用途 ?

/dev/null ,從名稱上可以很顯然看出是一個空文件(寫入到/dev/null時全部丟失,讀取/dev/null時自己返回EOF),那么你會很疑惑,他到底有什么用途呢,請看下文聽我講解,可能你在很多腳本里看過 /dev/null,具體總結下幾種常見用途.


一、禁止標準輸出.

eg:?

? ? cat $filename >/dev/null ?

? ? # 文件內容丟失,不會輸出到標準輸出,.


二、禁止標準錯誤

eg:?

? ? rm $badname 2>/dev/null ?

? ? #刪除文件錯誤時,不會再有提示到終端,都丟到/dev/null里去了


三、禁止標準輸出和標準錯誤的輸出.

eg1:?

? ? cat $filename 2>/dev/null >/dev/null?

? ? # 如果"$filename"不存在,將不會有任何錯誤信息提示.

? ? # 如果"$filename"存在, 文件的內容不會打印到標準輸出.?

? ? # 因此, 上面的代碼根本不會輸出任何信息.?

? ? # 當只想測試命令的退出碼而不想有任何輸出時非常有用。 ?

eg2:

? ? #-----------測試命令的退出 begin ----------------------# ?

? ? ls dddd 2>/dev/null 8?

? ? echo $?? ? //輸出命令退出代碼:0為命令正常執行,1-255為有出錯。 ?

? ? #-----------測試命令的退出 end-----------# ? ?

? ? cat $filename &>/dev/null?


四、清除日志文件內容

eg:

? ? cat /dev/null > /var/log/messages?

? ? #? : > /var/log/messages ? 有同樣的效果, 但不會產生新的進程.(因為:是內建的)? ?

? ? cat /dev/null > /var/log/wtmp


五、 隱藏cookie而不再使用

eg:

? ? if [ -f ~/.netscape/cookies ]? # 如果存在則刪除. ?

? ? then

? ? ? ? rm -f ~/.netscape/cookies ?

? ? fi

? ? ln -s /dev/null ~/.netscape/cookies?

? ? # 現在所有的cookies都會丟棄而不會保存在磁盤上了.


linuxshell變量$#,$@,$0,$1,$2的含義解釋:
變量說明:
$$
Shell本身的PIDProcessID
$!
Shell最后運行的后臺ProcessPID
$?
最后運行的命令的結束代碼(返回值)
$-
使用Set命令設定的Flag一覽
$*
所有參數列表。如"$*"用「"」括起來的情況、以"$1 $2 … $n"的形式輸出所有參數。
$@
所有參數列表。如"$@"用「"」括起來的情況、以"$1" "$2" … "$n"的形式輸出所有參數。
$#
添加到Shell的參數個數
$0
Shell本身的文件名
$1$n
添加到Shell的各參數值。$1是第1參數、$2是第2參數?




【shell】常用語法 -b file -c file -f file-d file -x file?



?shell】常用語法 -b file? -c file? -f file-d file? -x file? ? ?

一、test條件判斷

1test文件測試:

-b file ? ? 若文件存在且是一個塊特殊文件,則為真

-c file ? ? 若文件存在且是一個字符特殊文件,則為真

-d file ? ? 若文件存在且是一個目錄,則為真

-e file ? ? 若文件存在,則為真

-f file ? ? 若文件存在且為一個規則文件,則為真

-g file ? ? 若文件存在且設置了SGID位的值,則為真

-h file ? ? 若文件存在且為一個符號鏈接,則為真

-k file ? ? 若文件存在且設置了“sticky”位的值,則為真

-p file ? ? 若文件存在且為一已命名管道,則為真

-s file ? ? 若文件存在且其大小大于零,則為真

-u file ? ? 若文件存在且設置了SUID位的值,則為真

-r file ? ? 若文件存在且可讀,則為真

-w file ? ? 若文件存在且可寫,則為真

-x file ? ? 若文件存在且可執行,則為真

-o file ? ? 若文件存在且被有效用戶ID所擁有,則為真

?

2test字符串比較

-z string ? ? ? ? string長度為0,則為真

-n string ? ? ? ? string長度不為0,則為真

string1 = string2? ? 若兩個字符串相等,則為真

string1 != string2 ? 若兩個字符串不相等,則為真

?

3test命令的數字比較操作符

int1 -eq int2? ? ? int1等于int2,則為真

int1 –ne int2? ? ? int1不等于int2,則為真

int1 –lt int2? ? ? int1小于int2,則為真

int1 –le int2? ? ? int1小于等于int2,則為真

int1 –gt int2? ? ? int1大于int2,則為真

int1 –ge int2? ? ? int1大于等于int2,則為真

?

4test復合表達式

! expr ? ? ? ? ? ? ? ? ? ? expr為假則復合表達式為真。expr可以是任何有效的測試表達式

expr1 -a expr2 ? ? ? ? ? expr1expr2都為真,則為真

expr1 -o expr2 ? ? ? ? ? expr1expr2有一個為真,則為真

?

二、shell里的特殊變量

# sh /usr/local/bkeep/shell.sh 001 002

I'm $0 is:/usr/local/bkeep/shell.sh? ? ? ? //正在被執行腳本的名字;`basename $0`

I'm $1 is:001? ? ? ? ? ? ? ? //$1接收到的參數

I'm $2 is:002? ? ? ? ? ? ? ? //$2接收到的參數

I'm $# is:2 ? ? ? ? ? ? ? ? ? //總共接收到的參數個數

I'm $* is:001 002 ? ? ? ? //把接收到的參數全部打印出來

I'm $@ is:001 002? ? ? ? //同上

I'm $? is:0 ? ? ? ? ? ? ? ? ? //上一個腳本的退出狀態“0”代表正常;“1”非正常退出

I'm $$ is:24137? ? ? ? ? ? //當前執行腳本的進程ID

I'm $! is: ? ? ? ? ? ? ? ? ? ? //前一個后臺進程的id

?

:其他

1.返回值 #?

cd dir

echo "$?"

如果dir存在,則返回0,不存在則返回1.其他函數的命令也是一樣,比如diff,等等。

?

每個命令都會返回一個 exit 狀態(有時候也叫return狀態).成功返回0,如果返回一個非0,
常情況下都會被認為是一個錯誤碼.一個編寫良好的UNIX命令,程序,和工具都會返回一個0作為
退出碼來表示成功,雖然偶爾也會有例外.

同樣的,腳本中的函數和腳本本身都會返回退出狀態.在腳本或者是腳本函數中執行的最后的命
令會決定退出狀態.在腳本中,exit nnn命令將會把nnn 退出碼傳遞給shell

?

$?讀取最后執行命令的退出碼.函數返回后,$?給出函數最后執行的那條命令的退出碼.這種給
函數返回值的方法是 Bash 的方法.對于腳本來說也一樣.總之,一般情況下,0為成功,0失敗W

?

?2 .

$ 變量替換操作符
只有在變量被聲明,賦值,unsetexported 或者是在變量代表一個signal的時候,
變量才會是以本來的面目出現在腳本里.變量在被賦值的時候,可能需要使用"=",
read
狀態或者是在循環的頭部.
""中還是會發生變量替換,這被叫做部分引用,或叫弱引用.而在''中就不會發生變
量替換,這叫做全引用,也叫強引用.這兩個符合在同一個按鍵上。具體見第5章的討論.
注意:$var ${var}的區別,不加{},在某些上下文將引起錯誤,為了安全,使用2.

?

還有一個按鍵在左上角,即按鍵1的左邊。此按鍵有下面的作用。即將ls -l的結果傳給a

17 a=`ls -l` # ls -l的結果給a
18 echo $a # 別忘了,這么引用的話,ls的結果中的所有空白部分都沒了(包括換行)
19 echo
20 echo "$a" # 這么引用就正常了,保留了空白

?或者直接

echo "$(ls -l)" 也可以取出目錄

?

3.

read 命令狀態中
30 echo -n "Enter \"a\" "
31 read a
32 echo "The value of \"a\" is now $a."
33
34 echo
35

這個方法可以接受鍵盤輸入到a中。

?

?4.

在一個雙引號中直接使用變量名,一般都是沒有問題的.它阻止了所有在引號中的特殊字符的
重新解釋--包括變量名[2]--但是$,`\除外.[3]保留$,作為特殊字符的意義,是為了能夠在雙
引號中也能夠正常地引用變量("$var").這樣在""中可以使用變量所表達的值(Example 4-1).
使用""來防止單詞分割.[4]如果在參數列表中使用雙引號,將使得雙引號中的參數作為一個參
.即使雙引號中的字符串包含多個單詞(也就是包含空白部分),也不會變為多個參數,:
1 variable1="a variable containing five words"
2 COMMAND This is $variable1 # COMMAND 將以7 個參數來執行
3 # "This" "is" "a" "variable" "containing" "five" "words"
4
5 COMMAND "This is $variable1" # COMMAND 將以1 個參數來執行

?

5.賦值的方法

--------------------------------------------------------

aC風格的賦值
7 (( a = 23 )) # 給一個變量賦值,"="兩邊的空格就能看出這是c風格的處理.
8 echo "a (initial value) = $a"
9
10 (( a++ )) # 變量'a'后加1,C風格.
11 echo "a (after a++) = $a"
12
13 (( a-- )) # 變量'a'后減1,C風格.
14 echo "a (after a--) = $a"

?------------------------------------------------

37(( t = a<45?7:11 )) # C 風格的3 元操作.
38 echo "If a < 45, then t = 7, else t = 11."
39 echo "t = $t " # Yes!

---------------------------------------------------

18 LIMIT=10
19
20 for ((a=1; a <= LIMIT ; a++)) # Double parentheses, and "LIMIT" with no "$".
20 for ((a=1; a <= LIMIT ; a++)) # 雙圓括號, 并且"LIMIT"變量前邊沒有 "$".
21 do
22 echo -n "$a "
23 done

--------------------------------------------

b:let

4 a=2334 # 整型
5 let "a += 1" ? #不加雙引號也可以
6 echo "a = $a "? # a = 2335

--------------------------------------------------------------------

c:取變量

5 a=2334

6 a=$(($a+1))

7 echo "$a" ? #a=2335

?—————————————————————



本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/383331.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/383331.shtml
英文地址,請注明出處:http://en.pswp.cn/news/383331.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

拷貝構造函數的總結

構造函數的分類及調用 按照參數分類 1.無參構造&#xff08;默認構造&#xff09; 2.有參構造按照類型分類 1.普通構造函數2.拷貝構造函數無參構造寫法和調用 Person p1; 注意不能寫Person (),因為編譯器認為這個是函數聲明有參構造函數寫法 和調用 Person p2(10) 或者Per…

技術與技巧札記

Linux常用命令及技巧&#xff1a; &#xff08;1&#xff09;cat /proc/version 查看當前內核的版本 (2) 掛載nfs文件夾&#xff1a;需要先確認在&#xff0f;etc&#xff0f;exports文件&#xff0c;可以用于開發板掛載的文件夾 mount -o nolock 10.0.22.30:/root/sharednfs …

c++中new的總結(動態管理,malloc存在的問題,malloc與new的區別)

c中使用malloc出現的問題 程序員必須確定對象的長度malloc 返回一個&#xff08;void *&#xff09;指針 &#xff0c;c不允許將&#xff08;void*) 賦值給其它指針&#xff0c;必須強轉malloc可能申請內存失敗&#xff0c;所以必須判斷返回值來保存內存分配成功用戶在使用對象…

Linux中變量#,@,0,1,2,*,$$,$?的含義

$# 是傳給腳本的參數個數 $0 是腳本本身的名字 $1 是傳遞給該shell腳本的第一個參數 $2 是傳遞給該shell腳本的第二個參數 $ 是傳給腳本的所有參數的列表 $* 是以一個單字符串顯示所有向腳本傳遞的參數&#xff0c;與位置變量不同&#xff0c;參數可超過9個 $$ 是腳本運行的當前…

Volatile的陷阱

最近寫的關于在嵌入式開發中常遇到的關于volatile關鍵字使用的短文&#xff0c;都是些通用的技術&#xff0c;貼上來share。 對于volatile關鍵字&#xff0c;大部分的C語言教材都是一筆帶過&#xff0c;并沒有做太過深入的分析&#xff0c;所以這里簡單整理了一些關于volatile的…

c++中靜態成員變量和靜態成員函數

靜態成員變量 在一個類中&#xff0c;若將一個成員變量聲明為static,這種成員成為靜態成員變量&#xff0c;與一般的數據成員不同&#xff0c;無論建立了多少個對象&#xff0c;都只想有一個靜態數據的拷貝&#xff0c;靜態成員變量&#xff0c;屬于某個類&#xff0c;所有對象…

單列模式(餓漢)

單例模式案例 目的&#xff1a;為了讓類中只有一個實例&#xff0c;實例不需要自己釋放將 默認構造 和 拷貝構造 私有化內部維護一個 對象的指針私有化唯一指針對外提供getinstance方法來訪問這個指針保證類中只能實例化唯一 一個對象 主席案例 #include<iostream>usin…

Makefile札記

Makefile中: ? 的區別 在Makefile中我們經常看到 : ? 這幾個賦值運算符&#xff0c;那么他們有什么區別呢&#xff1f;我們來做個簡單的實驗 新建一個Makefile&#xff0c;內容為&#xff1a; ifdef DEFINE_VRE VRE “Hello World!” else endif ifeq ($(OPT),define) VRE…

c++中this指針基本概念和使用

class Person { public:int m_A;//非靜態成員變量&#xff0c;屬于對象上void func(/*Person * this*/){}; //非靜態成員函數 不屬于對象身上static int m_B;//靜態成員函數&#xff0c;不屬于對象上static void fun2(){};//靜態成員函數 &#xff0c;不屬于對象身上//double …

通用Makefile實現

Makefile是Linux下程序開發的自動化編譯工具&#xff0c;一個好的Makefile應該準確的識別編譯目標與源文件的依賴關系&#xff0c;并且有著高效的編譯效率&#xff0c;即每次重新make時只需要處理那些修改過的文件即可。Makefile擁有很多復雜的功能&#xff0c;這里不可能也沒必…

c++中空指針訪問成員函數

如果成員函數沒有用到this &#xff0c;那么空指針可以直接訪問 如果成員函數用到this 指針&#xff0c;就要注意&#xff0c;要判斷是否為空&#xff0c;防止程序崩潰 #include<iostream>using namespace std;class Person{public:void show(){//沒有 用到this指針&am…

從0開始python學習-35.allure報告企業定制

目錄 1. 搭建allure環境 2. 生成報告 3. logo定制 4. 企業級報告內容或層級定制 5. allure局域網查看 1. 搭建allure環境 1.1 JDK&#xff0c;使用PyCharm 找到pycharm安裝目錄找到java.exe記下jbr目錄的完整路徑&#xff0c;eg: C:\Program Files\JetBrains\PyCharm Com…

grep 常用命令

這個--include選項,可以這樣使用: grep -rn --include*.c --include*.h re . 可以指定多次, 如果真是上面的這種情況, 其實可以用 grep -rn --include*.[ch] re . 但是, 如果源文件中含有C源代碼,上面的方法就不湊效了, 因為[]中只能放一個字符. grep -rn --include*.{cp…

c++中友元函數詳解

友元 友元分為&#xff1a;友元函數和友元類 友元提供了一種突破封裝的方式&#xff0c;有時提供了便利。但是友元會增加耦合度&#xff0c;破壞了封裝&#xff0c;所以友元不宜多 用。 全局函數做友元函數 全局函數寫到類中做聲明 并且最前面寫關鍵字 friend 友元函數可訪問…

Linux時間函數札記

關于gmtime、gmtime_r、localtime、localtime_r 測試環境&#xff1a;vmware 7 Redhat5.5&#xff0c;系統時間使用UTC&#xff0c;時區為上海。 1、函數功能介紹 使用man gmtime或man localtime都可以的得到這幾個函數的介紹。原型如下&#xff1a; struct tm *gmtime(const …

c++實現順序表的相關操作

Myarray.h文件 #pragma once#include<iostream>using namespace std;class MyArray { public:MyArray();//默認構造 默認100容量MyArray(int capacity);MyArray(const MyArray& array);~MyArray();//尾插法void Push_Back(int val);//根據索引獲取值int getData(int…

系統架構札記

什么是高內聚、低耦合&#xff1f; 起因&#xff1a;模塊獨立性指每個模塊只完成系統要求的獨立子功能&#xff0c;并且與其他模塊的聯系最少且接口簡單&#xff0c;兩個定性的度量標準――耦合性和內聚性。 耦合性也稱塊間聯系。指軟件系統結構中各模塊間相互聯系緊密程度的一…

c++中運算符重載(加號運算,左移運算,前置后置++運算符,賦值運算,關系運算,函數運算)

運算符重載注意 重載的運算符要易讀內置的數據類型的表達式的運算符是不可以改變的不要重載&& 和 | | 運算符&#xff0c;[]和->運算符只能通過成員函數進行重載<<和>>只能通過全局函數配合友元函數進行重載 加號運算符重載 如果想讓自定義數據類型 進…

linux fstab解讀

fstab這個文件挺有用的。 從左到右&#xff1a; /dev/device mount-point type rules dump fsck 1. /dev/device: 不用說了吧&#xff1f;例如&#xff0c;/dev/hda1為M$-Win9x下的c:盤。 2. mount-point: 掛載點。例如&#xff0c;把/dev/hda1掛到/mnt/mywinc下。 3. type: ex…

c++實現字符串類的封裝

MyString.h文件 #define _CRT_SECURE_NO_WARNINGS#pragma once#include<iostream>#include<string>using namespace std;class MyString{friend ostream & operator<<(ostream & cout, MyString & str);friend istream & operator>>(…