我有這個 powershell 腳本,我想將檔案從我們的生產服務器復制到我們的測驗服務器,并且只復制更新或更改的檔案。腳本本身具有更多功能,但這種復制就是其中之一。我為此使用了以下 XCopy 命令:
... some other scriptingcode
try {
xCopy $source $destination /f /e /r /k /y /d
}
catch {
Write-Host "issue: $PSItem" -ForegroundColor Yellow -BackgroundColor Red
}
... some more other scriptingcode
當我XCopy直接從 powershell 提示符運行命令時,它會顯示它的進度,正在復制的檔案,而它正在運行,但是當我從腳本本身運行它時,它不會輸出任何內容。只是-Verbose在最后添加只會給我一個腳本錯誤。
我嘗試了以下方法:
$output = $(xCopy $source $destination /f /e /r /k /y /d) -join "`r\`n"
Write-Host $output
但這僅在已經復制后才給我輸出,并且在沒有復制 1 或 2 個檔案時不是問題,但當您有 100 或 1.000 個檔案要復制時并沒有真正的幫助。僅供參考,這-join "'r'n"是因為當輸出多于一行時,輸出是一個陣列,所以我不得不將這些行連接成一個字串,可以由Write-Host.
我試圖用谷歌搜索解決方案,但我想我的谷歌技能還不夠好,無法讓我得到這個問題的任何結果。那么,有沒有辦法在執行時XCopy輸出,就像直接從 powershell 命令列輸出一樣?
uj5u.com熱心網友回復:
對于xcopy,您既可以輸出到主機,也可以使用 捕獲輸出foreach:
try {
$output = xcopy $src $dst /f /e /r /k /y /d |
Foreach {
# write line to console
$_ | Write-Host -ForegroundColor Yellow -BackgroundColor Red
# add line to $output
$_
}
}
請注意,這xcopy不是 powershell 命令,因此它無法執行某些操作,例如使用默認值-verbose或-ErrorAction標志。它也不會引發終止錯誤,因此除非您正確設定,否則Try/Catch將不起作用$ErrorActionPreference。
uj5u.com熱心網友回復:
你不應該看到的問題與你的腳本,因為在默認情況下PowerShell的傳遞從輸出和錯誤輸出外部程式,如xcopy.exe 通過直接到主機(控制臺),而不與輸出或其時間干擾-無論你是否執行程式的互動式或從腳本。
類似地,如果您通過管道發送 stdout(成功)輸出,PowerShell 會對其進行流式處理,即在接收到的時候發送每個輸出行(可以想象,給定的程式可能會在不列印到控制臺時在發出它們之前自己緩沖行,但是似乎不是這種情況xcopy.exe)。
以流方式將外部程式的輸出列印到控制臺同時捕獲其輸出的最簡單方法是使用Tee-Objectcmdlet:
# Prints the output to the console as it is being received
# while also capturing it in variable $output
xcopy $source $destination /f /e /r /k /y /d | Tee-Object -Variable output
之后,$output包含 stdout 輸出作為行陣列(或單個字串,如果碰巧只有一個輸出行)。
要同時捕獲stderr輸出,請將重定向 附加2>&1到外部程式呼叫;stderr 行被捕獲為System.Management.Automation.ErrorRecord實體,就好像它們是PowerShell錯誤一樣;之后要將所有陣列元素轉換為字串,請使用$output = [string[]] $output.
字符編碼注意事項:當 PowerShell 參與處理外部程式輸出(在管道中,在變數中捕獲,保存到檔案)時,輸出總是首先解碼為 .NET 字串,基于存盤在 中的編碼[Console]::OutputEncoding,這可能根據情況必須修改以匹配給定外部程式使用的特定編碼 -有關更多資訊,請參閱此答案。
如果您想對從外部程式接收到的行執行轉換或應用其他格式,請考慮Cpt.Whale基于 ForEach-Object.
uj5u.com熱心網友回復:
這是$()在此處使用子運算式運算子的副作用(請注意,使用 group-expression 運算子也會獲得相同的效果())。發生這種情況的原因是因為$()與()功能等在數學括號,并且操作的順序(順序技術上的優先級在計算)決定了內表達外運算式之前制定。因此,括號內任何必須完成第一,然后才能由外碼處理。
在您的情況下,“外部代碼”是向控制臺顯示資訊的內容,因此,為什么內部命令必須在顯示之前完成。
如果您$output稍后不需要評估,最簡單和最高效的方法是將您的命令輸出直接傳送到Out-Host(您可能還想重定向錯誤輸出):
# 2>&1 will redirect the error stream (external command STDERR gets written here)
# to the success stream (external command STDOUT gets written here)
# The success stream is what gets passed down the pipeline
xCopy $source $destination /f /e /r /k /y /d 2>&1 | Out-Host
但是,如果您需要同時分配$output給一個變數以進行進一步處理,請參閱@Cpt.Whale 的解決方案,其中涉及巧妙地使用ForEach-Object. 他們的方法有一些邊緣警告,但確實允許您顯示輸出并在另一個變數中暫存輸出以供以后處理或在讀入時處理輸出行。@mklement0 的答案還顯示了如何使用Tee-Object同步輸出到變數和成功流。
轉載請註明出處,本文鏈接:https://www.uj5u.com/qianduan/338522.html
