• <label id="pxtpz"><meter id="pxtpz"></meter></label>
      1. <span id="pxtpz"><optgroup id="pxtpz"></optgroup></span>

        當前位置:雨林木風下載站 > 技術開發教程 > 詳細頁面

        如何在VB中截獲shell程序的輸出

        如何在VB中截獲shell程序的輸出

        更新時間:2022-05-04 文章作者:未知 信息來源:網絡 閱讀次數:

        在Windows環境下的所謂shell程序就是dos命令行程序,比如VC的CL.exe命令行編譯器,JDK的javac編譯器,啟動java程序用的java.exe都是標準的shell程序。截獲一個shell程序的輸出是很有用的,比如說您可以自己編寫一個IDE(集成開發環境),當用戶發出編譯指令時候,你可以在后臺啟動shell 調用編譯器并截獲它們的輸出,對這些輸出信息進行分析后在更為友好的用戶界面上顯示出來。為了方便起見,我們用VB作為本文的演示語言。

        通常,系統啟動Shell程序時缺省給定了3個I/O信道,標準輸入(stdin), 標準輸出stdout, 標準錯誤輸出stderr。之所以這么區分是因為在早期的計算機系統如PDP-11的一些限制。那時沒有GUI, 將輸出分為stdout,stderr可以避免程序的調試信息和正常輸出的信息混雜在一起。

        通常, shell程序把它們的輸出寫入標準輸出管道(stdout)、把出錯信息寫入標準錯誤管道(stderr)。缺省情況下,系統將管道的輸出直接送到屏幕,這樣一來我們就能看到應用程序運行結果了。

        為了捕獲一個標準控制臺應用程序的輸出,我們必須把standOutput和standError管道輸出重定向到我們自定義的管道。

        下面的代碼可以啟動一個shell程序,并將其輸出截獲。
        '執行并返回一個命令行程序(shell程序)的標準輸出和標準錯誤輸出'通常命令行程序的所有輸出都直接送到屏幕上Private Function ExecuteApp(sCmdline As String) As String Dim proc As PROCESS_INFORMATION, ret As Long Dim start As STARTUPINFO Dim sa As SECURITY_ATTRIBUTES Dim hReadPipe As Long '負責讀取的管道 Dim hWritePipe As Long '負責Shell程序的標準輸出和標準錯誤輸出的管道 Dim sOutput As String '放返回的數據 Dim lngBytesRead As Long, sBuffer As String * 256 sa.nLength = Len(sa) sa.bInheritHandle = True ret = CreatePipe(hReadPipe, hWritePipe, sa, 0) If ret = 0 Then MsgBox "CreatePipe failed. Error: " & Err.LastDllError Exit Function End If start.cb = Len(start) start.dwFlags = STARTF_USESTDHANDLES Or STARTF_USESHOWWINDOW' 把標準輸出和標準錯誤輸出重定向到同一個管道中去。start.hStdOutput = hWritePipe start.hStdError = hWritePipe start.wShowWindow = SW_HIDE ’隱含shell程序窗口 ' 啟動shell程序, sCmdLine指明執行的路徑 ret = CreateProcessA(0&, sCmdline, sa, sa, True, NORMAL_PRIORITY_CLASS, _ 0&, 0&, start, proc) If ret = 0 Then MsgBox "無法建立新進程,錯誤碼:" & Err.LastDllError Exit Function End If ' 本例中不必向shell程序送信息,因此可以先關閉hWritePipe CloseHandle hWritePipe ' 循環讀取shell程序的輸出,每次讀取256個字節。 Do ret = ReadFile(hReadPipe, sBuffer, 256, lngBytesRead, 0&) sOutput = sOutput & Left$(sBuffer, lngBytesRead) Loop While ret <> 0 ' 如果ret=0代表沒有更多的信息需要讀取了 ' 釋放相關資源 CloseHandle proc.hProcess CloseHandle proc.hThread CloseHandle hReadPipe ExecuteApp = sOutput ' 輸出結果End Function

        我對這個程序進行一些解釋。

        ret = CreatePipe(hReadPipe, hWritePipe, sa, 0)

        大家可以看到,首先我們建立一個匿名管道。該匿名管道稍候將用來取得與被截獲的應用程序的聯系。其中hReadPipe用來獲取shell程序的輸出,而hWritePipe可以用來向應用程序發送信息。如同現實世界中的水管一樣,水從管道的一端流進從另一端流出。您把水想象為信息,水管就是匿名管道,這樣一來就很好理解這段程序了。
        然后就是設置shell應用程序的初始屬性。 Dwflags可以指示系統在創建新進程時新進程使用了自定義的wShowWindow, hStdInput,hStdOutput和hStdError。(windows顯示屬性,標準輸入,標準輸出,標準錯誤輸出。)
        再把shell應用程序的標準輸出和標準錯誤輸出都定向到我們預先建好的管道中。
        代碼如下:

        start.dwFlags = STARTF_USESTDHANDLES Or STARTF_USESHOWWINDOW
        start.hStdOutput = hWritePipe
        start.hStdError = hWritePipe
        好,現在可以調用建立新進程的函數了:
        ret = CreateProcessA(0&, sCmdline, sa, sa, True, NORMAL_PRIORITY_CLASS, 0&, 0&, start, proc)
        然后,循環讀管道里的數據直到無數據可讀為止。
        Do
        ret = ReadFile(hReadPipe, sBuffer, 256, lngBytesRead, 0&) '每次讀256字節
        sOutput = sOutput & Left$(sBuffer, lngBytesRead) '送入一個字符串中
        Loop While ret <> 0 '若 ret = 0 表明沒有數據等待讀取。
        然后,釋放不用的資源。

        用法很簡單:比如:
        MsgBox ExecuteApp("c:\windows\command\mem.exe)

        是很方便吧?
        不過,這些程序是在NT下的,如果要在95下實現還需要一點點改動。因為如果該函數調用一個純win32的程序,沒問題。可是95是16,win32混合的系統,當你試圖調用一個16位的DOS應用程序那么,那么這個辦法會導致相關進程掛起。因為這涉及到WindowsNT和Windows 95對shell的不同實現。
        在win95中,16位shell程序關閉時并不保證重定向的管道也關閉,這樣,當你的程序試圖讀取一個已經關閉的shell程序的重定向管道時,你的程序就掛了。
        那么,有解決辦法嗎?回答是肯定的。
        解決辦法就是用一個win32的應用程序作為您的應用程序和shell程序的中間人。中間人程序繼承并重定向了主程序的輸入輸出,然后中間人程序啟動指定的shell程序。該shell程序也就繼承并重定向了主程序的輸入輸出。中間人程序一直等到shell程序結束才結束。
        當shell程序結束時,中間人程序也結束,同時因為中間人程序是一個win32程序,那么它就會關閉相應的重定向了管道。這樣,你的程序可以發現管道已經關閉,便可以跳出循環。你的程序就不會掛起了。
        下面是相關的中間人程序C代碼的實現:

        #include <windows.h>#include <stdio.h>void main (int argc, char *argv[]){ BOOL bRet = FALSE; STARTUPINFO si = {0}; PROCESS_INFORMATION pi = {0}; // Make child process use this app's standard files. si.cb = sizeof(si); si.dwFlags = STARTF_USESTDHANDLES; si.hStdInput = GetStdHandle (STD_INPUT_HANDLE); si.hStdOutput = GetStdHandle (STD_OUTPUT_HANDLE); si.hStdError = GetStdHandle (STD_ERROR_HANDLE); bRet = CreateProcess (NULL, argv[1], NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi ); if (bRet) { WaitForSingleObject (pi.hProcess, INFINITE); CloseHandle (pi.hProcess); CloseHandle (pi.hThread); }}
        把該程序編譯為conspawn.exe并放在系統可以調用到的路徑目錄中。
        然后把文章開頭提到的代碼中的CreateProcessA語句改為:
        ret = CreateProcessA(0&, "conspawn """ & sCmdline & """", sa, sa, True,
        NORMAL_PRIORITY_CLASS, 0&, 0&, start, proc)
        好,這樣一來,我們這個函數可以同時很好的支持WindowsNT和Windows95/98了。 

        溫馨提示:喜歡本站的話,請收藏一下本站!

        本類教程下載

        系統下載排行

        主站蜘蛛池模板: 亚洲国产成人久久笫一页| 免费视频专区一国产盗摄| 亚洲精品色婷婷在线影院 | 亚洲AV日韩AV鸥美在线观看| fc2免费人成在线视频| 亚洲中文字幕丝袜制服一区| 一级毛片a女人刺激视频免费| 亚洲男人第一无码aⅴ网站| 精品免费久久久久国产一区 | 国产亚洲精彩视频| 免费一级毛片在线播放不收费| 老司机午夜性生免费福利 | 亚洲一区二区在线视频| 最近免费中文字幕大全高清大全1 最近免费中文字幕mv在线电影 | 亚洲伊人久久大香线焦| 一二三四免费观看在线视频中文版| 亚洲免费一级视频| 麻豆国产VA免费精品高清在线| 国产成人亚洲精品无码AV大片| 国产黄色一级毛片亚洲黄片大全| 丁香花在线观看免费观看图片| 亚洲成色999久久网站| a拍拍男女免费看全片| 亚洲av成人无码网站… | 无码乱人伦一区二区亚洲| 亚洲美女免费视频| 亚洲AV性色在线观看| 国产性爱在线观看亚洲黄色一级片| 国产免费一区二区视频| 亚洲AV无码专区在线亚 | 亚洲精品国产成人专区| 毛片免费观看网站| japanese色国产在线看免费| 亚洲视频2020| 全黄性性激高免费视频| 亚洲免费在线视频| 小说区亚洲自拍另类| 亚洲视频在线视频| 亚洲成A人片在线观看中文| 久久免费观看国产精品| 久久亚洲精品无码gv|