問題:
在VC下想運行DOS命令,比如copy、dir、del等內部命令
是想直接運行這些命令,而不是調用相同功能的API函數
我的運行方法如下:
STARTUPINFO stInfo;
PROCESS_INFORMATION piInfo;
memset(&stInfo,0,sizeof(stInfo));
memset(&piInfo,0,sizeof(piInfo));
stInfo.cb=sizeof(STARTUPINFO);
GetStartupInfo(&stInfo);
char szCmd[1000] = {"del c://a.txt"};
int iCode = CreateProcess(NULL,szCmd,NULL,NULL,FALSE,CREATE_NO_WINDOW,//CREATE_NEW_CONSOLE,
NULL,NULL,&stInfo, &piInfo);
iCode = WinExec(szCmd,SW_HIDE);
iCode = (int)ShellExecute (NULL,"open",szCmd,NULL,NULL,SW_HIDE);
三種方法都不行,都提示找不到文件名。
我猜想可能是找不到DOS內部命令對應的文件的緣故
如何直接調用DOS下的命令而不用任何處理就執行呢?
解決方案:
使用WinExec命令
⑴ 函數原型:UINT WinExec(LPCSTR lpCmdLine, UINT uCmdShow);
⑵ 參數:lpCmdLine:指向一個空結束的字符串,串中包含將要執行的應用程序的命令行。
uCmdShow:定義Windows應用程序的窗口如何顯示,併爲CreateProcess函數提供STARTUPINFO參數的wShowWindow成員的值。
⑶ 返回值:
若函數調用成功,則返回值大於31。若函數調用失敗,則返回值爲下列之一:
① 0:系統內存或資源已耗盡。
② ERROR_BAD_FORMAT:EXE文件無效。
③ ERROR_FILE_NOT_FOUND:指定的文件未找到。
④ ERROR_PATH_NOT_FOUND:指定的路徑未找到。
雖然Microsoft認爲WinExec已過時,但是在許多時候,簡單的WinExec函數仍是運行新程序的最好方式。簡單地傳送作爲第一個參數的 命令行,還需要決定如何顯示程序的第二個參數。通常,將其設置爲SW_SHOW,也可嘗試SW_MINIMIZED或SW_MAXIMIZED。 WinExec不允許用CreateProcess獲得的所有選項,而它的確簡單。
3 使用ShellExecute命令
⑴ 函數原型:
HINSTANCE ShellExecute(HWND hwnd, LPCTSTR lpOperation, LPCTSTR lpFile, LPCTSTR lpParameters, LPCTSTR lpDirectory, INT nShowCmd);
⑵ 參數:
hwnd:指向父窗口的窗口句柄。此窗口接收應用程序產生的任何信息框。
lpOperation:一個空結束的字符串地址,此字符串指定要執行的操作。下面的操作字符串是有效的:
"open"
此函數打開由參數lpFile指定的文件,此文件可以是一個可執行文件或文檔文件,也可是一個文件夾。
"print"
此函數打印由參數lpFile指定的文件,此文件應是一個文檔文件,假如此文件是一個可執行文件,則打開此文件。
"explore"
此函數搜索由參數lpFile指定的文件夾,此文件應是一個文檔文件,
此參數可以爲空。這種情況下,函數用於打開由參數lpFile指定的文件。
lpFile:一個空結束的字符串地址,此字符串指定要打開或打印的文件或者是要打開或搜索的文件夾。
lpParameters:假如參數lpFile指定一個可執行文件,lpParameters則是一個空結束的字符串地址,此字符串指定要傳遞給應用程序的參數。假如lpFile指定一個文檔文件,lpParameters應爲空。
lpDirectory:一個空結束的字符串地址,此字符串指定默認目錄。
nShowCmd:假如lpFile指定一個可執行文件,nShowCmd表明應用程序打開時如何顯示。假如lpFile指定一個文檔文件,nShowCmd應爲空。
⑶ 返回值:
若函數調用成功,則返回值大於32,否則爲一個小於等於32的錯誤值。
說明:可以用此函數打開或搜索一個外殼文件夾。打開文件夾可用下面任何一種形式:
ShellExecute(handle, NULL, path_to_folder, NULL, NULL, SW_SHOWNORMAL);
或
ShellExecute(handle, "open", path_to_folder, NULL, NULL, SW_SHOWNORMAL);
搜索文件夾,可用如下形式
ShellExecute(handle, "explore", path_to_folder, NULL, NULL, SW_SHOWNORMAL);
ShellExecute命令雖已過時但易於得到。該命令向命令解釋程序提出打開、瀏覽或打印文檔或文件夾的請求,雖然可以用 ShellExecute運行程序,但通常只發送文檔名,而命令解釋程序則決定要運行那個程序。另外在打開目錄文件夾時,ShellExecute命令非 常有用。
⑷ 程序示例
下面通過一個例子來說名WinExec和ShellExecute的使用。下面程序有控制檯程序示例,其使用兩種不同的方法,打開文本文件。下面程序使用WinExec,並明確指定使用記事本程序。然後,使用ShellExecute,打開文本文件。
程序清單
#include
#include
void main(int argc,char *argv[])
{
cout<<"Opening with WinExec/n";
if (WinExec("notepad readme.txt",SH_SHOW)<32)
MessagBox(NULL,"Cant WinExec",NULL,MB_OK);
cout<<"Press Enter/n";
MessagBox(NULL,"Press OK to continue","Progrm Launched",MB_OK);
cout<<"Opening with ShellExecute/n";
if (ShellExecute (NULL,"open",
"readme.txt",NULL,NULL,SW_SHOW)<(HANDLE) 32)
MessagBox(NULL,"Cant ShellExecute/n",NULL,MB_OK);
}
4 使用CreateProcess命令
⑴ 函數原型:
BOOL CreateProcess(
LPCTSTR lpApplicationName,
LPTSTR lpCommandLine,
LPSECURITY_ATTRIBUTES lpProcessAttributes,
LPSECURITY_ATTRIBUTES lpThreadAttributes,
BOOL bInheritHandles,
DWORD dwCreationFlags,
LPVOID lpEnvironment,
LPCTSTR lpCurrentDirectory,
LPSTARTUPINFO lpStartupInfo,
LPPROCESS_INFORMATION lpProcessInformation
);
⑵ 參數:
lpApplicationName:指向一個以空結尾的串,他指定了要執行的模塊
lpCommandLine:指向一個以空結尾的串,該串定義了要執行的命令行。
lpProcessAttributes:指向一個SECURITY_ATTRIBUTES結構,該結構決定了返回的句柄是否可被子進程繼承。
lpThreadAttributes:指向一個SECURITY_ATTRIBUTES結構,該結構決定了返回的句柄是否可被子進程繼承。
bInheritHandles,:表明新進程是否從調用進程繼承句柄。
dwCreationFlags:定義控制優先類和進程創建的附加標誌。
lpEnvironment:指向一個新進程的環境塊。
lpCurrentDirectory:指向一個以空結尾的串,該串定義了子進程的當前驅動器和當前目錄。
lpStartupInfo:指向一個STARTUPINFO結構,該結構定義了新進程的主窗口將如何顯示。
lpProcessInformation:指向PROCESS_INFORMATION結構,該結構接受關於新進程的表示信息。
⑶ 返回值:
若函數調用成功,則返回值不爲0;若函數調用失敗,返回值爲0。
ShellExecute和WinExec命令用於簡單的作業。如果要完全控制一個新進程,就必須調用CreateProcess。
在上述參數中,參數lpStartupInfo是STARTUPINFO結構。可以用來設置控臺的標題,新窗口的的初始大小和位置,及重定向標準輸入 和輸出。新程序通常可以忽略多數這些數據項,如果選擇那樣做的話。可以規定該結構體中的標誌,已表明要設置的數據段。有時,不想設置任何信息,也必須傳遞 一個有效的指針給空結構。參數lpProcessInformation返回進程和線程句柄,還包括進程和線程ID。這些句柄擁有在參數 lpProcessAttributes和lpThreadAttributes中規定的訪問。
要注意,針對CreateProcess的一些參數對控制檯應用程序是特定的,而其它參數則對各種應用程序有用。大多數情況下,並不一定要填入 STARTUPINFO結構,但無論如何必須提供它。其返回值是布爾型的,而真正感興趣的返回值發生於作爲參數傳送的結構中。CreateProcess 返回該結構中的進程ID及其句柄,以及初始線程ID及其句柄。可以將ID發送到其它進程,或使用句柄來控制新進程。
=================================================
void runcmd(CString cmd, char *buf, unsigned int bufsize, unsigned int delay)
{
SECURITY_ATTRIBUTES sa;
HANDLE hRead, hWrite;
sa.nLength = sizeof(SECURITY_ATTRIBUTES);
sa.lpSecurityDescriptor = NULL;
sa.bInheritHandle = TRUE;
if (!CreatePipe(&hRead, &hWrite, &sa, 0)) {
return;
}
STARTUPINFO si;
PROCESS_INFORMATION pi;
si.cb = sizeof(STARTUPINFO);
GetStartupInfo(&si);
si.hStdError = hWrite;
si.hStdOutput = hWrite;
si.wShowWindow = SW_HIDE;
si.dwFlags = STARTF_USESHOWWINDOW | STARTF_USESTDHANDLES;
if (!CreateProcess(NULL, (LPTSTR)(LPCTSTR)cmd, NULL, NULL, TRUE, NULL, NULL, NULL, &si, &pi)) {
return;
}
CloseHandle(hWrite);
Sleep(delay);
DWORD bytesRead;
while (TRUE) {
memset(buf, 0, bufsize);
/* If the ReadFile function succeeds, the return value is nonzero. */
if (ReadFile(hRead, buf, bufsize, &bytesRead, NULL) != NULL) {
break;
}
Sleep(200);
}
CloseHandle(hRead);
return;
}
void CYUVviewerDlg::ExecDosCmd()
{
// TODO: Add your control notification handler code here
CString cmd = "ipconfig";
char *buf;
unsigned int bufsize = 1024;
unsigned int delay = 0;
if ((buf = (char *)malloc(bufsize)) == NULL) {
return;
}
memset(buf, 0, bufsize);
runcmd(cmd, buf, bufsize, delay);
//m_opt.Format("%s", buf);
UpdateData(FALSE);
free(buf);
}