2012-07-05 10 views
12

Mein Ziel ist es, eine externe ausführbare Datei in meinem Programm auszuführen. Zuerst habe ich system() Funktion verwendet, aber ich möchte nicht die Konsole für den Benutzer angezeigt werden. Also, ich suchte ein wenig und fand CreateProcess() Funktion. Wenn ich jedoch versuche, einen Parameter zu übergeben, weiß ich nicht warum, es schlägt fehl. Ich habe diesen Code von MSDN, und ein wenig verändert:CreateProcess() schlägt mit einer Zugriffsverletzung fehl

#include <windows.h> 
#include <stdio.h> 
#include <tchar.h> 

void _tmain(int argc, TCHAR *argv[]) 
{ 
    STARTUPINFO si; 
    PROCESS_INFORMATION pi; 

    ZeroMemory(&si, sizeof(si)); 
    si.cb = sizeof(si); 
    ZeroMemory(&pi, sizeof(pi)); 
    /* 
    if(argc != 2) 
    { 
     printf("Usage: %s [cmdline]\n", argv[0]); 
     return; 
    } 
    */ 
    // Start the child process. 
    if(!CreateProcess(NULL, // No module name (use command line) 
     L"c:\\users\\e\\desktop\\mspaint.exe",  // Command line 
     NULL,   // Process handle not inheritable 
     NULL,   // Thread handle not inheritable 
     FALSE,   // Set handle inheritance to FALSE 
     0,    // No creation flags 
     NULL,   // Use parent's environment block 
     NULL,   // Use parent's starting directory 
     &si,   // Pointer to STARTUPINFO structure 
     &pi)   // Pointer to PROCESS_INFORMATION structure 
    ) 
    { 
     printf("CreateProcess failed (%d).\n", GetLastError()); 
     return; 
    } 

    // Wait until child process exits. 
    WaitForSingleObject(pi.hProcess, INFINITE); 

    // Close process and thread handles. 
    CloseHandle(pi.hProcess); 
    CloseHandle(pi.hThread); 
} 

jedoch dieser Code irgendwie Zugriffsverletzung crated. Kann ich mspaint ausführen, ohne dem Benutzer die Konsole anzuzeigen?

Vielen Dank.

+2

Für eine Sache, 'CreateProcess' erfordert, dass sein zweiter Parameter ein nicht-const string (falls vorhanden) sein. Ich bin mir nicht sicher, ob das in der Praxis ein Problem ist, aber ich wollte es der Vollständigkeit halber erwähnen. – reuben

+0

... und wo sonst passiert AV? Hast du einen Call-Stack? – reuben

+0

@reuben Uhm ... Ich bin mir nicht ganz sicher, aber ich denke, dies ist die Ausgabe von Call-Stack: 'kernel32.dll! 76da70ac() \t \t [Frames unten möglicherweise falsch und/oder fehlen, keine Symbole geladen für kernel32.dll] \t > \t msvcr100d.dll! _nh_malloc_dbg (unsigned int nSize, int nhFlag, int nBlockUse, const char * szFileName, int nLine) Linie 302 + 0x1D \t C++ ' –

Antwort

7

Versuchen Sie es, es sollte funktionieren.

TCHAR lpszClientPath[500]= TEXT("c:\\users\\e\\desktop\\mspaint.exe"); 
if(!CreateProcess(NULL, lpszClientPath, NULL, NULL, FALSE, NORMAL_PRIORITY_CLASS|CREATE_NEW_CONSOLE|CREATE_UNICODE_ENVIRONMENT,NULL, NULL, &si, &pi)) 
      { 
    printf("CreateProcess failed (%d).\n", GetLastError()); 
     return; 
      } 
... 
... 
18

Das zweite Argument ist ein LPTSTR, nämlich ein Zeiger auf ein nicht-const Char-Array. Die docs ausdrücklich sagen:

diese Parameter nicht ein Zeiger auf Nur-Lese-Speicher (wie ein const Variable oder eine Zeichenkette) sein können

Der Grund Führen einen Stringliteral ist ein Problem:

Das System fügt der Befehlszeile ein abschließendes Nullzeichen hinzu, um den Dateinamen von den Argumenten zu trennen. Dies teilt die ursprüngliche Zeichenfolge in zwei Zeichenfolgen für die interne Verarbeitung.

Das bedeutet in Ihrem Fall, es versucht, nur Lese-Speicher zu modifizieren, daher der Absturz.

+0

Bytes ich einen LPTSTR nicht zuordnen kann entweder . Was muss mein zweites Argument sein, wenn ich fragen darf, um mspaint auszuführen, das sich unter "c: \ users \ e \ desktop \" befindet? Vielen Dank. –

+2

@JohnDoe: 'wchar_t Pfad [] = L" C: \\ Benutzer \\ e \\ Desktop \\ "' – hmjd

+0

'wchar_t Pfad [] = L" C: \\ blahblah "' ist nicht das selbe wie 'LPWSTR path = L "C: \\ blahblah" 'ist aber dasselbe wie' wchar_t * path = L "C: \\ blahblah" '. Sie sollten entweder das verwenden, was @hmjd vorgeschlagen hat, oder Sie sollten 'LPWSTR path' definieren und dann Speicher für diese Variable zuweisen und den Pfad dorthin kopieren. –

0

ändern Sie den Code zu diesem:

#include <windows.h> 
#include <stdio.h> 
#include <tchar.h> 

void _tmain(int argc, TCHAR *argv[]) 
{ 
    TCHAR ProcessName[256]; 
    STARTUPINFO si; 
    PROCESS_INFORMATION pi; 

    wcscpy(ProcessName,L"c:\\users\\e\\desktop\\mspaint.exe"); 
    ZeroMemory(&si, sizeof(si)); 
    si.cb = sizeof(si); 
    ZeroMemory(&pi, sizeof(pi)); 
    /* 
    if(argc != 2) 
    { 
     printf("Usage: %s [cmdline]\n", argv[0]); 
     return; 
    } 
    */ 
    // Start the child process. 
    if(!CreateProcess(NULL, // No module name (use command line) 
     ProcessName,  // Command line 
     NULL,   // Process handle not inheritable 
     NULL,   // Thread handle not inheritable 
     FALSE,   // Set handle inheritance to FALSE 
     0,    // No creation flags 
     NULL,   // Use parent's environment block 
     NULL,   // Use parent's starting directory 
     &si,   // Pointer to STARTUPINFO structure 
     &pi)   // Pointer to PROCESS_INFORMATION structure 
    ) 
    { 
     printf("CreateProcess failed (%d).\n", GetLastError()); 
     return; 
    } 

    // Wait until child process exits. 
    WaitForSingleObject(pi.hProcess, INFINITE); 

    // Close process and thread handles. 
    CloseHandle(pi.hProcess); 
    CloseHandle(pi.hThread); 
}