PowerShell與Unix Shell對比:八大實例
本文將舉例對比PowerShell和Unix Shell,通常是Linux Bourne Shell(包括sh、ksh和bash等)。二者存在非常大的差異,***不同是PowerShell將對象作為基本的操作單元,而Unix Shell將字符串作為基本單元;相似之處是二者均有數量巨大內置命令,而且允許用戶擴展。
1 實例:終止進程
在Unix操作系統中為了終止所有以字母“p”開頭的進程,需要在命令行中運行下面的命令:
$ ps -e | grep " p" | awk '{ print $1 }' | xargs kill
通過ps命令獲取了當前進程的清單并將獲取的文本輸出到grep命令中,該命令搜索文件名以“p”開頭的進程。將輸出發送給awk命令,從中選取第1列(這里是進程的ID)并輸出給xargs命令;xargs命令會對每個進程執行kill命令,從而終止所有以“p”開頭的進程。盡管實現了功能,但是整個命令卻不可靠。因為ps命令的執行效果在不同操作系統中不同(甚至在相同的操作系統的不同版本中執行也會有差異)。如果不支持-e選項的ps在執行時包含進程ID的列,則不一定是第1列,此時命令行的執行會出現問題。
類似地,如果要在PowerShell中執行相同的命令,只需要執行下面的操作:
PS C:\> get-process p* | stop-process
這里的命令查找所有以“p”開頭的進程,并將其終止。Get-Process cmdlet帶的參數是需要匹配的進程名,得到的結果對象被直接傳遞給Stop-Process cmdlet,這樣即可結束對應的對象進程。
2 實例:結束過濾的進程
為查找并殺死占用內存超過10 MB的進程,在Unix命令行下需要執行如下命令:
$ ps -el | awk '{ if ( $6 > (1024*10)) { print $3 } }' |
grep -v PID | xargs kill
此命令的執行成功取決于用戶已知道ps –el命令將會在第6列中返回進程占用的內存大小(以KB為單位)并且在第3列中包含PID屬性,同時需要去掉ps命令輸出的第1行。
接下來查看在PowerShell中對應的腳本:
PS C:\> get-process | where { $_.WS -gt 10MB } | stop-process
可以看到基于對象的命令相比基于文字的命令的好處,即不必關心包含進程占用內存大小或包含進程ID的部分所在的列。內存占用量能夠通過進程名稱引用,Where cmdlet可檢查輸入的對象并取得對象的屬性。
3 實例:計算目錄大小
在這個實例中將計算某個目錄中包含文件的大小,遍歷文件、獲取其長度屬性并疊加到一個變量中,***打印變量。在Unix系統中的處理方式如下:
$ tot=0; for file in $( ls ) > do > set -- $( ls -log $file ) > echo $3 > (( tot = $tot + $3 )) > done; echo $tot
上例使用set命令為每個空格分隔的元素創建變量,這是在awk命令出現之前通常使用的Unix命令。如果使用了awk命令,將會減少相當的代碼量,如下所示:
$ ls –l | awk ‘{ tot += $5; print tot; }’ | tail -1
這樣降低了命令的復雜性,但是需要用戶知道其中的長度屬性是在第5列,不同版本的awk可能會有差異。在PowerShell中的循環也很簡單,雖然也需要逐個遍歷,但是獲取長度屬性很方便。因為長度作為文件對象的屬性存在,不需要關心其所在列,相似的腳本如下:
PS C:\> get-childitem | measure-object -Property length -Sum Count : 53 Average : Sum : 489648208 Maximum : Minimum : Property : length
其中使用的Measure-Object cmdlet將根據輸入的-Property確定要操作的屬性,根據輸入的選項,如-Sum、-Maximum、-Minimum和-Average對前面指定的對象屬性做求和、求***值、求最小值,以及求平均值。這里為了和前面的Unix腳本相匹配,只需要指定Length屬性及-Sum選項即可。
4 實例:操作動態值
很多由系統提供的對象通常是動態,而不是靜態的。即獲取某個對象后不需要稍后再次獲取其數據,因為該數據根據系統條件的改變不斷更新。并且修改這些對象也會立即在系統中生效,此類對象稱之為“實時對象”。
例如,需要獲取處理器處理時間的占用情況,傳統Unix Shell中的ps命令會反復運行以不斷取得進程的運行狀況。對于能夠訪問實時進程的對象,只需要獲取處理對象一次。一旦對象被系統更新,只需要持續重新讀取相同的屬性即可。下例以10秒為間隔獲取應用程序占用的內存大小,首先查看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 實例:監視進程壽命
如果在Unix下確定特定的進程是否繼續在運行,則需要收集進程清單并與另外一個清單對比。例如:
$ 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下只需要執行下面的操作:
PS C:\> $processToWatch = get-process applicationName PS C:\> $processToWatch.WaitForExit()
能夠看到PowerShell只需獲取對象并等待對象退出即可提示。
6 實例:確定軟件是否存在預發布的測試版
由于預發布的測試版本往往不穩定,可能存在各種缺陷。從而導致系統安全風險,所以有必要區分此類軟件。Unix可執行文件中并不保存此類信息,所以不必討論Unix Shell下的情況。對于Windows來說,需要特定的工具檢查當前運行的進程中是否存在預發布的測試版本。例如:
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是確定軟件是否為預發布版本的根據。如果該屬性為真,則Get-Process cmdlet輸出的對象將會被輸出到控制臺。
7 實例: 轉換字符串大小寫
在Unix Shell中通常會用下面的方式轉換特定字符串的大小寫:
$ 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()方法轉換將字符串中的大寫字母轉換為小寫字母。如果需要將大寫字母轉換為小寫字母,則使用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
【編輯推薦】























