PowerShell與Unix Shell對比:八大實例
本文將舉例對比PowerShell和Unix Shell,通常是Linux Bourne Shell(包括sh、ksh和bash等)。二者存在非常大的差異,***不同是PowerShell將對象作為基本的操作單元,而Unix Shell將字符串作為基本單元;相似之處是二者均有數(shù)量巨大內(nèi)置命令,而且允許用戶擴展。
1 實例:終止進程
在Unix操作系統(tǒng)中為了終止所有以字母“p”開頭的進程,需要在命令行中運行下面的命令:
$ ps -e | grep " p" | awk '{ print $1 }' | xargs kill
通過ps命令獲取了當(dāng)前進程的清單并將獲取的文本輸出到grep命令中,該命令搜索文件名以“p”開頭的進程。將輸出發(fā)送給awk命令,從中選取第1列(這里是進程的ID)并輸出給xargs命令;xargs命令會對每個進程執(zhí)行kill命令,從而終止所有以“p”開頭的進程。盡管實現(xiàn)了功能,但是整個命令卻不可靠。因為ps命令的執(zhí)行效果在不同操作系統(tǒng)中不同(甚至在相同的操作系統(tǒng)的不同版本中執(zhí)行也會有差異)。如果不支持-e選項的ps在執(zhí)行時包含進程ID的列,則不一定是第1列,此時命令行的執(zhí)行會出現(xiàn)問題。
類似地,如果要在PowerShell中執(zhí)行相同的命令,只需要執(zhí)行下面的操作:
PS C:\> get-process p* | stop-process
這里的命令查找所有以“p”開頭的進程,并將其終止。Get-Process cmdlet帶的參數(shù)是需要匹配的進程名,得到的結(jié)果對象被直接傳遞給Stop-Process cmdlet,這樣即可結(jié)束對應(yīng)的對象進程。
2 實例:結(jié)束過濾的進程
為查找并殺死占用內(nèi)存超過10 MB的進程,在Unix命令行下需要執(zhí)行如下命令:
$ ps -el | awk '{ if ( $6 > (1024*10)) { print $3 } }' |
grep -v PID | xargs kill
此命令的執(zhí)行成功取決于用戶已知道ps –el命令將會在第6列中返回進程占用的內(nèi)存大小(以KB為單位)并且在第3列中包含PID屬性,同時需要去掉ps命令輸出的第1行。
接下來查看在PowerShell中對應(yīng)的腳本:
PS C:\> get-process | where { $_.WS -gt 10MB } | stop-process
可以看到基于對象的命令相比基于文字的命令的好處,即不必關(guān)心包含進程占用內(nèi)存大小或包含進程ID的部分所在的列。內(nèi)存占用量能夠通過進程名稱引用,Where cmdlet可檢查輸入的對象并取得對象的屬性。
3 實例:計算目錄大小
在這個實例中將計算某個目錄中包含文件的大小,遍歷文件、獲取其長度屬性并疊加到一個變量中,***打印變量。在Unix系統(tǒng)中的處理方式如下:
$ tot=0; for file in $( ls ) > do > set -- $( ls -log $file ) > echo $3 > (( tot = $tot + $3 )) > done; echo $tot
上例使用set命令為每個空格分隔的元素創(chuàng)建變量,這是在awk命令出現(xiàn)之前通常使用的Unix命令。如果使用了awk命令,將會減少相當(dāng)?shù)拇a量,如下所示:
$ ls –l | awk ‘{ tot += $5; print tot; }’ | tail -1
這樣降低了命令的復(fù)雜性,但是需要用戶知道其中的長度屬性是在第5列,不同版本的awk可能會有差異。在PowerShell中的循環(huán)也很簡單,雖然也需要逐個遍歷,但是獲取長度屬性很方便。因為長度作為文件對象的屬性存在,不需要關(guān)心其所在列,相似的腳本如下:
PS C:\> get-childitem | measure-object -Property length -Sum Count : 53 Average : Sum : 489648208 Maximum : Minimum : Property : length
其中使用的Measure-Object cmdlet將根據(jù)輸入的-Property確定要操作的屬性,根據(jù)輸入的選項,如-Sum、-Maximum、-Minimum和-Average對前面指定的對象屬性做求和、求***值、求最小值,以及求平均值。這里為了和前面的Unix腳本相匹配,只需要指定Length屬性及-Sum選項即可。
4 實例:操作動態(tài)值
很多由系統(tǒng)提供的對象通常是動態(tài),而不是靜態(tài)的。即獲取某個對象后不需要稍后再次獲取其數(shù)據(jù),因為該數(shù)據(jù)根據(jù)系統(tǒng)條件的改變不斷更新。并且修改這些對象也會立即在系統(tǒng)中生效,此類對象稱之為“實時對象”。
例如,需要獲取處理器處理時間的占用情況,傳統(tǒng)Unix Shell中的ps命令會反復(fù)運行以不斷取得進程的運行狀況。對于能夠訪問實時進程的對象,只需要獲取處理對象一次。一旦對象被系統(tǒng)更新,只需要持續(xù)重新讀取相同的屬性即可。下例以10秒為間隔獲取應(yīng)用程序占用的內(nèi)存大小,首先查看Unix Shell腳本的處理方式:
$ while [ true ]
do
msize1=$(ps -el|grep application|grep -v grep|awk '{ print $6}')
sleep 10
msize2=$(ps -el|grep application|grep -v grep|awk '{print $6}')
expr $msize2 - $msize1
msize1=$msize2
done
使用PowerShell操作的方式如下:
PS C:\>> $app = get-process applicationName
PS C:\>> while ( $true ) {
>> $msize1 = $app.VM
>> start-sleep 10
>> $app.VM - $msize1
>> }
顯然,PowerShell的腳本更加簡單易讀。
5 實例:監(jiān)視進程壽命
如果在Unix下確定特定的進程是否繼續(xù)在運行,則需要收集進程清單并與另外一個清單對比。例如:
$ processToWatch=$( ps -e | grep application | awk '{ print $1 }'
$ while [ true ]
> do
> sleep 10
> processToCheck=$(ps -e |grep application |awk '{print $1}' )
> if [ -z "$processToCheck" -or \
> "$processToWatch" != "$processToCheck" ]
> then
> echo "Process application is not running"
> return
> fi
> done
而在PowerShell下只需要執(zhí)行下面的操作:
PS C:\> $processToWatch = get-process applicationName PS C:\> $processToWatch.WaitForExit()
能夠看到PowerShell只需獲取對象并等待對象退出即可提示。
6 實例:確定軟件是否存在預(yù)發(fā)布的測試版
由于預(yù)發(fā)布的測試版本往往不穩(wěn)定,可能存在各種缺陷。從而導(dǎo)致系統(tǒng)安全風(fēng)險,所以有必要區(qū)分此類軟件。Unix可執(zhí)行文件中并不保存此類信息,所以不必討論Unix Shell下的情況。對于Windows來說,需要特定的工具檢查當(dāng)前運行的進程中是否存在預(yù)發(fā)布的測試版本。例如:
PS C:\> Get-Process | where  {
>> $_.mainmodule.FileVersioninfo.isPreRelease}
>> 
Handles  NPM(K)    PM(K)      WS(K) VM(M)   CPU(s)     Id ProcessName
-------  ------    -----      ----- -----   ------     -- -----------
    627      17    29236       5120   177     7.66    560 Powerword
這個實例使用了層疊的屬性,從進程對象(MainModule)中獲取的屬性中檢查。FileVersionInfo屬性是MainModle的引用,其屬性IsPreRelease是確定軟件是否為預(yù)發(fā)布版本的根據(jù)。如果該屬性為真,則Get-Process cmdlet輸出的對象將會被輸出到控制臺。
7 實例: 轉(zhuǎn)換字符串大小寫
在Unix Shell中通常會用下面的方式轉(zhuǎn)換特定字符串的大小寫:
$ echo "this is a string" | tr [:lower:] [:upper:]
或者使用:
$ echo "this is a string" | tr '[a-z]' '[A.Z]'
在PowerShell中則更為簡便,如:
PS (1) > "this is a string".ToUpper()
這是直接使用字符串對象的ToUpper()方法轉(zhuǎn)換將字符串中的大寫字母轉(zhuǎn)換為小寫字母。如果需要將大寫字母轉(zhuǎn)換為小寫字母,則使用ToLower()方法即可。
8 實例:在字符串中插入字符
例如,需要將字符串“ABC”插入到字符串“string”的首字母之后,得到類似“sABCtring”的字符串。在Unix Shell下使用sed命令:
$ echo "string" | sed "s|\(.\)\(.*)|\1ABC\2|"
在PowerShell中則使用正則表達式:
PS (1) > "string" -replace '(.)(.*)','$1ABC$2' sABCtring
在PowerShell中更簡單的方法是對字符串對象使用Insert()方法直接完成插入操作,如:
PS (2) > "string".Insert(1,"ABC") sABCtring
【編輯推薦】















 
 
 






 
 
 
 