日本国产亚洲-日本国产一区-日本国产一区二区三区-日本韩国欧美一区-日本韩国欧美在线-日本韩国欧美在线观看

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

如何在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了。 

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

本類教程下載

系統下載排行

主站蜘蛛池模板: 久久www免费人成_看片美女图 | 91精品综合 | 做爰三级在线播放 | 日韩精品亚洲人成在线观看 | 国产精品视频免费观看 | 国产成人精品亚洲 | 国产精品私人玩物在线观看 | 欧美激情在线播放一区二区 | 青青草成人在线 | 在线免费观看色 | 久久成人精品免费播放 | 成年人网站免费在线观看 | 精品欧美一区二区三区精品久久 | 亚洲综合一二三区 | 中文字幕欧美日韩 | 久久久久久国产视频 | 亚洲欧美不卡视频 | 欧美在线观看成人高清视频 | 亚洲国产欧美在线人成北岛玲 | 中国国产成人精品久久 | 日韩在线视频不卡 | 91se在线| 大香线蕉伊人久久爱 | 可播放的免费男男videos不卡 | 精品国产一区二区三区久久 | 日皮视频免费看 | 国产精品综合久成人 | 四虎影院紧急入口 | 91视频最新网址 | 91日本在线 | 亚洲精品一区二区四季 | 中文字幕一区视频 | 人人澡人摸人人爽歪歪 | 日韩欧美国产精品第一页不卡 | 色综合久久综合欧美综合网 | 午夜寂寞成人国产全部排列表安卓 | 欧美日比 | 中文字幕国产在线观看 | 99久热只有精品视频免费看 | 欧美日韩亚洲在线观看 | 久久精品亚洲 |