web-dev-qa-db-fra.com

C++ Exécution des commandes CMD

J'ai un grave problème ici. J'ai besoin d'exécuter une ligne de commande CMD via C++ sans afficher la fenêtre de la console. Par conséquent, je ne peux pas utiliser system(cmd), car la fenêtre s’affiche.

J'ai essayé winExec(cmd, SW_HIDE), mais cela ne fonctionne pas non plus. CreateProcess est un autre que j'ai essayé. Cependant, cela concerne les programmes en cours ou les fichiers de commandes.

J'ai fini par essayer ShellExecute:

ShellExecute( NULL, "open",
    "cmd.exe",
    "ipconfig > myfile.txt",
    "c:\projects\b",
    SW_SHOWNORMAL
);

Quelqu'un peut-il voir quelque chose de mal avec le code ci-dessus? J'ai utilisé SW_SHOWNORMAL jusqu'à ce que je sache que cela fonctionne.

J'ai vraiment besoin d'aide avec ça. Rien n'est apparu et j'essaie depuis un certain temps. Tout conseil que quelqu'un pourrait donner serait génial :)

12
JP29

La redirection de la sortie vers votre propre canal est une solution plus simple, car elle évite de créer le fichier de sortie, mais cela fonctionne correctement:

ShellExecute(0, "open", "cmd.exe", "/C ipconfig > out.txt", 0, SW_HIDE);

Vous ne voyez pas la fenêtre cmd et le résultat est redirigé comme prévu.

Votre code est probablement en train d'échouer (hormis le /C) car vous spécifiez le chemin comme "c:\projects\b" plutôt que "c:\\projects\\b".

8
arx

Vous devez utiliser CreateProcess on cmd.exe avec le paramètre /C pour tunneler la commande ipconfig. Le> ne fonctionne pas en soi sur la ligne de commande. Vous devez rediriger par programme le stdout .

4
Felice Pollano

Voici mon implémentation d'une fonction DosExec qui permet d'exécuter (en silence) toute commande DOS et d'extraire la sortie générée sous forme de chaîne unicode.

// Convert an OEM string (8-bit) to a UTF-16 string (16-bit) 
#define OEMtoUNICODE(str)   CHARtoWCHAR(str, CP_OEMCP)

/* Convert a single/multi-byte string to a UTF-16 string (16-bit).
 We take advantage of the MultiByteToWideChar function that allows to specify the charset of the input string.
*/
LPWSTR CHARtoWCHAR(LPSTR str, UINT codePage) {
    size_t len = strlen(str) + 1;
    int size_needed = MultiByteToWideChar(codePage, 0, str, len, NULL, 0);
    LPWSTR wstr = (LPWSTR) LocalAlloc(LPTR, sizeof(WCHAR) * size_needed);
    MultiByteToWideChar(codePage, 0, str, len, wstr, size_needed);
    return wstr;
}

/* Execute a DOS command.

 If the function succeeds, the return value is a non-NULL pointer to the output of the invoked command. 
 Command will produce a 8-bit characters stream using OEM code-page.

 As charset depends on OS config (ex: CP437 [OEM-US/latin-US], CP850 [OEM 850/latin-1]),
 before being returned, output is converted to a wide-char string with function OEMtoUNICODE.

 Resulting buffer is allocated with LocalAlloc.
 It is the caller's responsibility to free the memory used by the argument list when it is no longer needed. 
 To free the memory, use a single call to LocalFree function.
*/
LPWSTR DosExec(LPWSTR command){
    // Allocate 1Mo to store the output (final buffer will be sized to actual output)
    // If output exceeds that size, it will be truncated
    const SIZE_T RESULT_SIZE = sizeof(char)*1024*1024;
    char* output = (char*) LocalAlloc(LPTR, RESULT_SIZE);

    HANDLE readPipe, writePipe;
    SECURITY_ATTRIBUTES security;
    STARTUPINFOA        start;
    PROCESS_INFORMATION processInfo;

    security.nLength = sizeof(SECURITY_ATTRIBUTES);
    security.bInheritHandle = true;
    security.lpSecurityDescriptor = NULL;

    if ( CreatePipe(
                    &readPipe,  // address of variable for read handle
                    &writePipe, // address of variable for write handle
                    &security,  // pointer to security attributes
                    0           // number of bytes reserved for pipe
                    ) ){


        GetStartupInfoA(&start);
        start.hStdOutput  = writePipe;
        start.hStdError   = writePipe;
        start.hStdInput   = readPipe;
        start.dwFlags     = STARTF_USESTDHANDLES + STARTF_USESHOWWINDOW;
        start.wShowWindow = SW_HIDE;

// We have to start the DOS app the same way cmd.exe does (using the current Win32 ANSI code-page).
// So, we use the "ANSI" version of createProcess, to be able to pass a LPSTR (single/multi-byte character string) 
// instead of a LPWSTR (wide-character string) and we use the UNICODEtoANSI function to convert the given command 
        if (CreateProcessA(NULL,                    // pointer to name of executable module
                           UNICODEtoANSI(command),  // pointer to command line string
                           &security,               // pointer to process security attributes
                           &security,               // pointer to thread security attributes
                           TRUE,                    // handle inheritance flag
                           NORMAL_PRIORITY_CLASS,   // creation flags
                           NULL,                    // pointer to new environment block
                           NULL,                    // pointer to current directory name
                           &start,                  // pointer to STARTUPINFO
                           &processInfo             // pointer to PROCESS_INFORMATION
                         )){

            // wait for the child process to start
            for(UINT state = WAIT_TIMEOUT; state == WAIT_TIMEOUT; state = WaitForSingleObject(processInfo.hProcess, 100) );

            DWORD bytesRead = 0, count = 0;
            const int BUFF_SIZE = 1024;
            char* buffer = (char*) malloc(sizeof(char)*BUFF_SIZE+1);
            strcpy(output, "");
            do {                
                DWORD dwAvail = 0;
                if (!PeekNamedPipe(readPipe, NULL, 0, NULL, &dwAvail, NULL)) {
                    // error, the child process might have ended
                    break;
                }
                if (!dwAvail) {
                    // no data available in the pipe
                    break;
                }
                ReadFile(readPipe, buffer, BUFF_SIZE, &bytesRead, NULL);
                buffer[bytesRead] = '\0';
                if((count+bytesRead) > RESULT_SIZE) break;
                strcat(output, buffer);
                count += bytesRead;
            } while (bytesRead >= BUFF_SIZE);
            free(buffer);
        }

    }

    CloseHandle(processInfo.hThread);
    CloseHandle(processInfo.hProcess);
    CloseHandle(writePipe);
    CloseHandle(readPipe);

    // convert result buffer to a wide-character string
    LPWSTR result = OEMtoUNICODE(output);
    LocalFree(output);
    return result;
}
4
Cédric Françoys

J'ai un programme similaire [Windows 7 et 10 testés] sur github 

https://github.com/vlsireddy/remwin/tree/master/remwin

Ceci est le programme du serveur qui 

  1. écoute l'interface nommée "Connexion au réseau local" dans Windows pour le port UDP (5555) et reçoit le paquet udp. 
  2. le contenu du paquet udp reçu est exécuté sur cmd.exe [veuillez ne pas fermer cmd.exe après avoir exécuté la commande et la chaîne de sortie [la sortie de la commande exécutée] est renvoyée au programme client par le même port udp].
  3. En d'autres termes, la commande Reçue dans un paquet udp -> un paquet udp analysé -> exécutée sur cmd.exe -> une sortie renvoyée sur le même port vers le programme client

Cela ne montre pas "fenêtre de console" Pas besoin de quelqu'un pour exécuter manuellement la commande sur cmd.exe Remwin.exe peut être exécuté en arrière-plan et c'est un programme de serveur léger

1
particlereddy