同时捕获和显示标准输出(STDOUT)

6
我有以下代码来捕获和处理Run命令的输出。我该如何修改它,使得Run命令窗口显示输出并同时将输出记录下来?将 @SW_HIDE 替换为 @SW_SHOW (或等效命令) 只会显示一个空白命令窗口。
与Linux中的tee命令类似,可以将STDOUT打印到文件中进行记录。
$CurrentPID = Run(@ComSpec & ' /c ' & $CurrentLogCmd, "", @SW_HIDE, $STDERR_CHILD + $STDOUT_CHILD)

If Not ProcessWaitClose($CurrentPID,60) Then
    WriteLog("[Warning] Timed-out.Finding date in current hour raw log -" & $CurrentLogFileName)
    $F_LogWarningExist = 1
    Return $C_SUCCESS ; Take chances and proceed with parsing raw logs
EndIf

$CurrentOutput = StdoutRead($CurrentPID)
2个回答

4
ConsoleWrite(_getDOSOutput('ipconfig /all') & @CRLF)

Func _getDOSOutput($command)
    Local $text = '', $Pid = Run('"' & @ComSpec & '" /c ' & $command, '', @SW_HIDE, 2 + 4)
    While 1
            $text &= StdoutRead($Pid, False, False)
            If @error Then ExitLoop
            Sleep(10)
    WEnd
    Return StringStripWS($text, 7)
EndFunc   ;==>_getDOSOutput

也许这会对您有所帮助。

1
我实际上需要文本在窗口中打印,除了重定向...这两者可以同时进行吗? - Siva
你是什么意思? - Xenobiologist
1
我希望将文本打印到屏幕上并重定向,以便AutoIt可以读取和解析。基本上,当我在监视器上时,我希望能即时查看脚本的进度情况,而不是等到AutoIt在结束时抛出错误(如果有的话)。 - Siva

1
类似于Linux中的tee命令,可以在打印标准输出的同时记录到文件中。重定向STDOUT使接收方(脚本)负责显示和记录。根据文档所述:

StdoutRead()不会阻塞,它将立即返回。为了获取所有数据,必须在循环中调用。

示例:
#AutoIt3Wrapper_Change2CUI=Y

#include <Constants.au3>
#include <MsgBoxConstants.au3>

Global Enum  $EXIT_OK, _
             $EXIT_NOCOMPILE, _
             $EXIT_ABORT

Global Const $g_sPromptError  = 'Compile this script and run resulting executable instead.', _
             $g_sPromptInput  = 'Enter a command:', _
             $g_sInputDefault = 'ping localhost -n 10'

Global       $g_sCMD = '', _
             $g_sSTD = ''

Main()

Func Main()

    If Not @Compiled Then

        MsgBox($MB_OK + $MB_ICONERROR, @ScriptName, $g_sPromptError)
        Exit $EXIT_NOCOMPILE

    EndIf

    $g_sCMD = InputBox(@ScriptName, $g_sPromptInput, $g_sInputDefault)
    If @error Then Exit $EXIT_ABORT

    $g_sSTD = _getCmdStd($g_sCMD)
    MsgBox($MB_OK + $MB_ICONINFORMATION, $g_sCMD, $g_sSTD)

    Exit $EXIT_OK
EndFunc

Func _getCmdStd(Const $sCMD, Const $sDir = '', Const $iType = $STDERR_MERGED, Const $bShow = False, Const $iDelay = 100)
    Local       $sTMP = ''
    Local       $sSTD = ''
    Local       $sCOM = @ComSpec & ' /c ' & $sCMD
    Local Const $iWin = $bShow ? @SW_SHOW : @SW_HIDE
    Local Const $iPID = Run($sCOM, $sDir, $iWin, $iType)

    While True

        $sTMP = StdoutRead($iPID, False, False)

        If @error Then

            ExitLoop 1

        ElseIf $sTMP Then

            $sTMP  = StringReplace($sTMP, @CR & @CR, '')
            $sSTD &= $sTMP

            ConsoleWrite($sTMP)

        EndIf

        Sleep($iDelay)

    WEnd

    Return SetError(@error, @extended, $sSTD)
EndFunc

执行完成后返回STDOUT(和STDERR),在执行期间写入控制台。根据需要替换MsgBox() (日志记录函数)。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接