ฉันจะรันคำสั่งและรับเอาต์พุตของคำสั่งภายใน C ++ โดยใช้ POSIX ได้อย่างไร


463

ฉันกำลังมองหาวิธีที่จะได้รับผลลัพธ์ของคำสั่งเมื่อมันถูกเรียกใช้จากภายในโปรแกรม C ++ ฉันได้ดูการใช้งานsystem()ฟังก์ชั่น แต่นั่นก็จะดำเนินการคำสั่ง นี่คือตัวอย่างของสิ่งที่ฉันกำลังมองหา:

std::string result = system("./some_command");

ฉันต้องการเรียกใช้คำสั่งโดยพลการและรับเอาต์พุต ฉันดูที่boost.orgแต่ฉันไม่พบสิ่งใดที่จะให้สิ่งที่ฉันต้องการ


โปรดดูคำตอบในคำถามนี้: /programming/52164723/how-to-execute-a-command-and-get-return-code-stdout-and-stderr-of-command-in-cสำหรับส่วนขยายของคำตอบที่ดีด้านล่างซึ่งให้วิธีการรับreturn codeและstderrรวมถึงstdoutคำตอบที่อธิบายไว้แล้ว
code_fodder

2
@code_fodder คุณสามารถสร้างลิงก์ไปยังstackoverflow.com/questions/52164723/…
Jonas Stein

คำตอบ:


599
#include <cstdio>
#include <iostream>
#include <memory>
#include <stdexcept>
#include <string>
#include <array>

std::string exec(const char* cmd) {
    std::array<char, 128> buffer;
    std::string result;
    std::unique_ptr<FILE, decltype(&pclose)> pipe(popen(cmd, "r"), pclose);
    if (!pipe) {
        throw std::runtime_error("popen() failed!");
    }
    while (fgets(buffer.data(), buffer.size(), pipe.get()) != nullptr) {
        result += buffer.data();
    }
    return result;
}

รุ่น Pre-C ++ 11:

#include <iostream>
#include <stdexcept>
#include <stdio.h>
#include <string>

std::string exec(const char* cmd) {
    char buffer[128];
    std::string result = "";
    FILE* pipe = popen(cmd, "r");
    if (!pipe) throw std::runtime_error("popen() failed!");
    try {
        while (fgets(buffer, sizeof buffer, pipe) != NULL) {
            result += buffer;
        }
    } catch (...) {
        pclose(pipe);
        throw;
    }
    pclose(pipe);
    return result;
}

แทนที่popenและpcloseด้วย_popenและ_pcloseสำหรับ Windows


69
โปรดทราบว่านี้จะคว้าstdoutและไม่stderr
kalaxy

14
นอกจากนี้โปรดทราบว่าอาจมีข้อยกเว้นเกิดขึ้นได้result += bufferดังนั้นท่ออาจไม่ถูกปิดอย่างเหมาะสม
Fred Foo

6
@Yasky: int main(){ puts("ERROR"); }เมื่อโปรแกรมที่กำลังดำเนินการคือ
dreamlax

8
คำตอบนั้นดี แต่มันจะดีกว่าถ้าคุณแทนที่ 'char * cmd' ด้วย 'const char * cmd'
fnc12

29
unique_ptr เป็นแบบที่ดีกว่าที่นี่ซึ่งไม่เคยใช้จำนวนการอ้างอิงจริง
Czipperz

77

รับทั้ง stdout และ stderr (และเขียนถึง stdin, ไม่แสดงที่นี่) เป็นเรื่องง่ายกับหัวข้อpstreamsของฉันซึ่งกำหนดคลาส iostream ที่ทำงานเช่นpopen:

#include <pstream.h>
#include <string>
#include <iostream>

int main()
{
  // run a process and create a streambuf that reads its stdout and stderr
  redi::ipstream proc("./some_command", redi::pstreams::pstdout | redi::pstreams::pstderr);
  std::string line;
  // read child's stdout
  while (std::getline(proc.out(), line))
    std::cout << "stdout: " << line << '\n';
  # if reading stdout stopped at EOF then reset the state:
  if (proc.eof() && proc.fail())
    proc.clear();
  // read child's stderr
  while (std::getline(proc.err(), line))
    std::cout << "stderr: " << line << '\n';
} 

18
ฉันไม่เห็นด้วย. popenคุณต้องการใช้ C stdio API ฉันต้องการ iostreams API popenคุณต้องล้างFILEด้ามจับด้วยตนเองส่วนpstreams ทำโดยอัตโนมัติ popenยอมรับเฉพาะconst char*อาร์กิวเมนต์ที่ต้องใช้ความระมัดระวังเพื่อหลีกเลี่ยงการโจมตีด้วยการฉีดเชลล์ pstreams ช่วยให้คุณสามารถส่งเวกเตอร์ของสตริงที่คล้ายกับexecvซึ่งปลอดภัยกว่า popenไม่ได้ให้อะไรนอกจากเป็นท่อ pstreams บอกคุณว่า PID ของเด็กอนุญาตให้คุณส่งสัญญาณเช่นจะฆ่ามันถ้ามันถูกบล็อคหรือไม่ออก ทั้งหมดนี้เป็นข้อดีแม้ว่าคุณจะต้องการเพียงทิศทางเดียว IO
Jonathan Wakely

1
ปัญหาอีกประการของการแก้ไขนี้คือถ้าเด็กเขียนไปยัง stderr เพียงพอที่จะเติมบัฟเฟอร์และบล็อกก่อนที่จะเริ่มเขียนไปยัง stdout ผู้ปกครองจะบล็อก stdout การอ่านในขณะที่เด็กถูกบล็อครอ stderr ที่จะอ่าน การหยุดชะงักของทรัพยากร! อย่างน้อยหนึ่งลูปเหล่านั้นน่าจะดีกว่าแบบอะซิงโครนัส (เช่นเธรด)
Jesse Chisholm

1
@JesseChisholm ใช่นั่นอาจเป็นปัญหา แต่คุณไม่จำเป็นต้องใช้เธรดเนื่องจาก pstreams อนุญาตให้ประมาณของการไม่บล็อก I / O โดยใช้อินเทอร์เฟซ iostream โดยเฉพาะการใช้ฟังก์ชันreadsomeซึ่งตรวจสอบความพร้อมในการใช้pstreambuf::in_avail()ดังนั้นจะไม่บล็อก ที่ช่วยให้ demultiplexing ในกระบวนการ 'stdout และ stderr เนื่องจากแต่ละรายการมีข้อมูลที่พร้อมใช้งาน pstreambuf::in_avail()ใช้งานได้อย่างน่าเชื่อถือ 100% เท่านั้นหากระบบปฏิบัติการรองรับ FIONREAD ioctl ที่ไม่ได้มาตรฐาน แต่ได้รับการสนับสนุนบน (อย่างน้อย) GNU / Linux และ Solaris
โจนาธาน Wakely

13
@chiliNUT รุ่นใหม่ 1.0.1 ใช้สิทธิ์การใช้งาน Boost
Jonathan Wakely

1
@Janathan ฉันจะฆ่า ipstream ได้อย่างไรหลังจากพูดว่าหมดเวลา 5 วินาที
AK

34

ฉันต้องการใช้popen () (++ waqas)

แต่บางครั้งคุณต้องอ่านและเขียน ...

ดูเหมือนว่าไม่มีใครทำสิ่งที่ยากอีกต่อไป

(สมมติว่าเป็นสภาพแวดล้อม Unix / Linux / Mac หรือ Windows ที่มีเลเยอร์ความเข้ากันได้ POSIX ... )

enum PIPE_FILE_DESCRIPTERS
{
  READ_FD  = 0,
  WRITE_FD = 1
};

enum CONSTANTS
{
  BUFFER_SIZE = 100
};

int
main()
{
  int       parentToChild[2];
  int       childToParent[2];
  pid_t     pid;
  string    dataReadFromChild;
  char      buffer[BUFFER_SIZE + 1];
  ssize_t   readResult;
  int       status;

  ASSERT_IS(0, pipe(parentToChild));
  ASSERT_IS(0, pipe(childToParent));

  switch (pid = fork())
  {
    case -1:
      FAIL("Fork failed");
      exit(-1);

    case 0: /* Child */
      ASSERT_NOT(-1, dup2(parentToChild[READ_FD], STDIN_FILENO));
      ASSERT_NOT(-1, dup2(childToParent[WRITE_FD], STDOUT_FILENO));
      ASSERT_NOT(-1, dup2(childToParent[WRITE_FD], STDERR_FILENO));
      ASSERT_IS(0, close(parentToChild [WRITE_FD]));
      ASSERT_IS(0, close(childToParent [READ_FD]));

      /*     file, arg0, arg1,  arg2 */
      execlp("ls", "ls", "-al", "--color");

      FAIL("This line should never be reached!!!");
      exit(-1);

    default: /* Parent */
      cout << "Child " << pid << " process running..." << endl;

      ASSERT_IS(0, close(parentToChild [READ_FD]));
      ASSERT_IS(0, close(childToParent [WRITE_FD]));

      while (true)
      {
        switch (readResult = read(childToParent[READ_FD],
                                  buffer, BUFFER_SIZE))
        {
          case 0: /* End-of-File, or non-blocking read. */
            cout << "End of file reached..."         << endl
                 << "Data received was ("
                 << dataReadFromChild.size() << "): " << endl
                 << dataReadFromChild                << endl;

            ASSERT_IS(pid, waitpid(pid, & status, 0));

            cout << endl
                 << "Child exit staus is:  " << WEXITSTATUS(status) << endl
                 << endl;

            exit(0);


          case -1:
            if ((errno == EINTR) || (errno == EAGAIN))
            {
              errno = 0;
              break;
            }
            else
            {
              FAIL("read() failed");
              exit(-1);
            }

          default:
            dataReadFromChild . append(buffer, readResult);
            break;
        }
      } /* while (true) */
  } /* switch (pid = fork())*/
}

คุณอาจต้องการเล่นโดยเลือก () และไม่บล็อกการอ่าน

fd_set          readfds;
struct timeval  timeout;

timeout.tv_sec  = 0;    /* Seconds */
timeout.tv_usec = 1000; /* Microseconds */

FD_ZERO(&readfds);
FD_SET(childToParent[READ_FD], &readfds);

switch (select (1 + childToParent[READ_FD], &readfds, (fd_set*)NULL, (fd_set*)NULL, & timeout))
{
  case 0: /* Timeout expired */
    break;

  case -1:
    if ((errno == EINTR) || (errno == EAGAIN))
    {
      errno = 0;
      break;
    }
    else
    {
      FAIL("Select() Failed");
      exit(-1);
    }

  case 1:  /* We have input */
    readResult = read(childToParent[READ_FD], buffer, BUFFER_SIZE);
    // However you want to handle it...
    break;

  default:
    FAIL("How did we see input on more than one file descriptor?");
    exit(-1);
}

1
วิธีที่ยากลำบากคือถูก :) ฉันชอบความคิดที่มีการโทรแบบเลือก () แต่ในกรณีนี้ฉันต้องรอจนกว่างานจะเสร็จสมบูรณ์ ฉันจะเก็บรหัสนี้สำหรับโครงการอื่นฉันมี :)
Misha M

4
... หรือคุณสามารถใช้ฟังก์ชั่น posix_spawnp ที่มีอยู่
ต่อ Johansson

5
การexeclpโทรของคุณมีข้อบกพร่อง: argตัวชี้สุดท้ายที่ส่งผ่านจะต้อง(char *) NULLถูกยกเลิกรายการอาร์กิวเมนต์ Variadic อย่างถูกต้อง (ดูexeclp(3)สำหรับการอ้างอิง)
Kristóf Marussy

มันจะทำงานบนยูนิกซ์, ลินุกซ์และหน้าต่างได้หรือไม่? คุณสามารถช่วยหัวไฟล์ด้วยได้ไหม?
kittu

คุณจะส่งไฟล์. bat ในรหัสไปที่ใด
kittu

33

สำหรับ Windows popenยังใช้งานได้ แต่จะเปิดหน้าต่างคอนโซลซึ่งกะพริบผ่านแอปพลิเคชัน UI ของคุณอย่างรวดเร็ว หากคุณต้องการเป็นมืออาชีพควรปิดการใช้งาน "กะพริบ" นี้ (โดยเฉพาะอย่างยิ่งหากผู้ใช้สามารถยกเลิกได้)

ดังนั้นนี่คือเวอร์ชั่นของฉันสำหรับ Windows:

(รหัสนี้ถูกรวมใหม่บางส่วนจากแนวคิดที่เขียนในตัวอย่างโครงการและตัวอย่าง MSDN)

#include <windows.h>
#include <atlstr.h>
//
// Execute a command and get the results. (Only standard output)
//
CStringA ExecCmd(
    const wchar_t* cmd              // [in] command to execute
)
{
    CStringA strResult;
    HANDLE hPipeRead, hPipeWrite;

    SECURITY_ATTRIBUTES saAttr = {sizeof(SECURITY_ATTRIBUTES)};
    saAttr.bInheritHandle = TRUE; // Pipe handles are inherited by child process.
    saAttr.lpSecurityDescriptor = NULL;

    // Create a pipe to get results from child's stdout.
    if (!CreatePipe(&hPipeRead, &hPipeWrite, &saAttr, 0))
        return strResult;

    STARTUPINFOW si = {sizeof(STARTUPINFOW)};
    si.dwFlags     = STARTF_USESHOWWINDOW | STARTF_USESTDHANDLES;
    si.hStdOutput  = hPipeWrite;
    si.hStdError   = hPipeWrite;
    si.wShowWindow = SW_HIDE; // Prevents cmd window from flashing.
                              // Requires STARTF_USESHOWWINDOW in dwFlags.

    PROCESS_INFORMATION pi = { 0 };

    BOOL fSuccess = CreateProcessW(NULL, (LPWSTR)cmd, NULL, NULL, TRUE, CREATE_NEW_CONSOLE, NULL, NULL, &si, &pi);
    if (! fSuccess)
    {
        CloseHandle(hPipeWrite);
        CloseHandle(hPipeRead);
        return strResult;
    }

    bool bProcessEnded = false;
    for (; !bProcessEnded ;)
    {
        // Give some timeslice (50 ms), so we won't waste 100% CPU.
        bProcessEnded = WaitForSingleObject( pi.hProcess, 50) == WAIT_OBJECT_0;

        // Even if process exited - we continue reading, if
        // there is some data available over pipe.
        for (;;)
        {
            char buf[1024];
            DWORD dwRead = 0;
            DWORD dwAvail = 0;

            if (!::PeekNamedPipe(hPipeRead, NULL, 0, NULL, &dwAvail, NULL))
                break;

            if (!dwAvail) // No data available, return
                break;

            if (!::ReadFile(hPipeRead, buf, min(sizeof(buf) - 1, dwAvail), &dwRead, NULL) || !dwRead)
                // Error, the child process might ended
                break;

            buf[dwRead] = 0;
            strResult += buf;
        }
    } //for

    CloseHandle(hPipeWrite);
    CloseHandle(hPipeRead);
    CloseHandle(pi.hProcess);
    CloseHandle(pi.hThread);
    return strResult;
} //ExecCmd

1
นี่เป็นทางออกโปรดของฉันสำหรับ Windows ฉันหวังว่าคุณจะยกโทษให้การเปลี่ยนแปลงของฉัน ฉันอยากจะแนะนำให้นักแสดงเพิ่มชัดเจนมากขึ้นในขณะที่ฉันพิจารณาการใช้งานที่ชัดเจนwchar_tและCreateProcessWเป็นข้อ จำกัด ที่ไม่จำเป็น
Wolf

คุณเห็นปัญหาหรือปัญหาที่อาจเกิดขึ้นกับนักแสดงนี้หรือไม่? ฉันชอบที่จะเก็บรหัสอย่างน้อยและอย่าเขียนโดยไม่จำเป็น
TarmoPikaro

4
หลังจากอ่านฟังก์ชั่น CreateProcess (Windows)ฉันเห็นอันตรายที่แท้จริงในการทำเช่นนี้The Unicode version of this function, CreateProcessW, can modify the contents of this string. Therefore, this parameter cannot be a pointer to read-only memory (such as a const variable or a literal string). If this parameter is a constant string, the function may cause an access violation.ดังนั้นจึงเป็นการดีกว่าที่จะคัดลอกบรรทัดคำสั่งลงในบัฟเฟอร์ที่แยกต่างหากก่อนเพื่อป้องกันไม่ให้ผู้เรียกเปลี่ยนอินพุตต้นฉบับ
Wolf

คำตอบนี้จัดการ stderr ไม่ถูกต้อง
Refael Sheinker

สิ่งนี้ใช้ได้กับระบบ Unix หรือไม่ หรือฉันต้องใช้อย่างอื่นสำหรับอุปกรณ์ Unix?
255.tar.xz

17

สองแนวทางที่เป็นไปได้:

  1. ฉันไม่คิดว่าpopen()เป็นส่วนหนึ่งของมาตรฐาน C ++ (เป็นส่วนหนึ่งของ POSIX จากหน่วยความจำ) แต่มันมีอยู่ใน UNIX ทุกตัวที่ฉันทำงานด้วย (และดูเหมือนว่าคุณจะกำหนดเป้าหมาย UNIX ตั้งแต่คำสั่งของคุณ./some_command)

  2. ในกรณีที่ไม่มีโอกาสpopen()คุณสามารถใช้system("./some_command >/tmp/some_command.out");จากนั้นใช้ฟังก์ชัน I / O ปกติเพื่อประมวลผลไฟล์เอาต์พุต


ขอบคุณสำหรับ popen ฉันจะใช้มันในตอนนี้และฉันจะกังวลเกี่ยวกับระบบที่ไม่ใช่ POSIX หากเกิดขึ้น
Misha M

8

ฉันไม่สามารถเข้าใจได้ว่าทำไม popen / pclose หายไปจากCode :: Blocks / MinGW ดังนั้นฉันจึงแก้ไขปัญหาโดยใช้ CreateProcess () และ CreatePipe () แทน

นี่คือวิธีแก้ปัญหาสำหรับฉัน:

//C++11
#include <cstdio>
#include <iostream>
#include <windows.h>
#include <cstdint>
#include <deque>
#include <string>
#include <thread>

using namespace std;

int SystemCapture(
    string         CmdLine,    //Command Line
    string         CmdRunDir,  //set to '.' for current directory
    string&        ListStdOut, //Return List of StdOut
    string&        ListStdErr, //Return List of StdErr
    uint32_t&      RetCode)    //Return Exit Code
{
    int                  Success;
    SECURITY_ATTRIBUTES  security_attributes;
    HANDLE               stdout_rd = INVALID_HANDLE_VALUE;
    HANDLE               stdout_wr = INVALID_HANDLE_VALUE;
    HANDLE               stderr_rd = INVALID_HANDLE_VALUE;
    HANDLE               stderr_wr = INVALID_HANDLE_VALUE;
    PROCESS_INFORMATION  process_info;
    STARTUPINFO          startup_info;
    thread               stdout_thread;
    thread               stderr_thread;

    security_attributes.nLength              = sizeof(SECURITY_ATTRIBUTES);
    security_attributes.bInheritHandle       = TRUE;
    security_attributes.lpSecurityDescriptor = nullptr;

    if (!CreatePipe(&stdout_rd, &stdout_wr, &security_attributes, 0) ||
            !SetHandleInformation(stdout_rd, HANDLE_FLAG_INHERIT, 0)) {
        return -1;
    }

    if (!CreatePipe(&stderr_rd, &stderr_wr, &security_attributes, 0) ||
            !SetHandleInformation(stderr_rd, HANDLE_FLAG_INHERIT, 0)) {
        if (stdout_rd != INVALID_HANDLE_VALUE) CloseHandle(stdout_rd);
        if (stdout_wr != INVALID_HANDLE_VALUE) CloseHandle(stdout_wr);
        return -2;
    }

    ZeroMemory(&process_info, sizeof(PROCESS_INFORMATION));
    ZeroMemory(&startup_info, sizeof(STARTUPINFO));

    startup_info.cb         = sizeof(STARTUPINFO);
    startup_info.hStdInput  = 0;
    startup_info.hStdOutput = stdout_wr;
    startup_info.hStdError  = stderr_wr;

    if(stdout_rd || stderr_rd)
        startup_info.dwFlags |= STARTF_USESTDHANDLES;

    // Make a copy because CreateProcess needs to modify string buffer
    char      CmdLineStr[MAX_PATH];
    strncpy(CmdLineStr, CmdLine.c_str(), MAX_PATH);
    CmdLineStr[MAX_PATH-1] = 0;

    Success = CreateProcess(
        nullptr,
        CmdLineStr,
        nullptr,
        nullptr,
        TRUE,
        0,
        nullptr,
        CmdRunDir.c_str(),
        &startup_info,
        &process_info
    );
    CloseHandle(stdout_wr);
    CloseHandle(stderr_wr);

    if(!Success) {
        CloseHandle(process_info.hProcess);
        CloseHandle(process_info.hThread);
        CloseHandle(stdout_rd);
        CloseHandle(stderr_rd);
        return -4;
    }
    else {
        CloseHandle(process_info.hThread);
    }

    if(stdout_rd) {
        stdout_thread=thread([&]() {
            DWORD  n;
            const size_t bufsize = 1000;
            char         buffer [bufsize];
            for(;;) {
                n = 0;
                int Success = ReadFile(
                    stdout_rd,
                    buffer,
                    (DWORD)bufsize,
                    &n,
                    nullptr
                );
                printf("STDERR: Success:%d n:%d\n", Success, (int)n);
                if(!Success || n == 0)
                    break;
                string s(buffer, n);
                printf("STDOUT:(%s)\n", s.c_str());
                ListStdOut += s;
            }
            printf("STDOUT:BREAK!\n");
        });
    }

    if(stderr_rd) {
        stderr_thread=thread([&]() {
            DWORD        n;
            const size_t bufsize = 1000;
            char         buffer [bufsize];
            for(;;) {
                n = 0;
                int Success = ReadFile(
                    stderr_rd,
                    buffer,
                    (DWORD)bufsize,
                    &n,
                    nullptr
                );
                printf("STDERR: Success:%d n:%d\n", Success, (int)n);
                if(!Success || n == 0)
                    break;
                string s(buffer, n);
                printf("STDERR:(%s)\n", s.c_str());
                ListStdOut += s;
            }
            printf("STDERR:BREAK!\n");
        });
    }

    WaitForSingleObject(process_info.hProcess,    INFINITE);
    if(!GetExitCodeProcess(process_info.hProcess, (DWORD*) &RetCode))
        RetCode = -1;

    CloseHandle(process_info.hProcess);

    if(stdout_thread.joinable())
        stdout_thread.join();

    if(stderr_thread.joinable())
        stderr_thread.join();

    CloseHandle(stdout_rd);
    CloseHandle(stderr_rd);

    return 0;
}

int main()
{
    int            rc;
    uint32_t       RetCode;
    string         ListStdOut;
    string         ListStdErr;

    cout << "STARTING.\n";

    rc = SystemCapture(
        "C:\\Windows\\System32\\ipconfig.exe",    //Command Line
        ".",                                     //CmdRunDir
        ListStdOut,                              //Return List of StdOut
        ListStdErr,                              //Return List of StdErr
        RetCode                                  //Return Exit Code
    );
    if (rc < 0) {
        cout << "ERROR: SystemCapture\n";
    }

    cout << "STDOUT:\n";
    cout << ListStdOut;

    cout << "STDERR:\n";
    cout << ListStdErr;

    cout << "Finished.\n";

    cout << "Press Enter to Continue";
    cin.ignore();

    return 0;
}

5
ขอบคุณ! นี่เป็นการใช้งาน popen ที่ดีที่สุดสำหรับ Windows บนอินเทอร์เน็ต! และเมื่อผ่านการตั้งค่าสถานะ CREATE_NO_WINDOW ในที่สุดก็สามารถกำจัดการแจ้งเตือน cmd ที่น่ารำคาญที่ปรากฏขึ้น
Lacho Tomov

1
คุณจะผ่านสิ่งที่อยู่CREATE_NO_WINDOWที่ไหน
Refael Sheinker

2
@Bill Moore ถ้าคุณสังเกตเห็นว่ามีข้อผิดพลาดในคำตอบของคุณ ListStdErrไม่เคยใช้
Refael Sheinker

7

ต่อไปนี้อาจเป็นโซลูชันแบบพกพา มันเป็นไปตามมาตรฐาน

#include <iostream>
#include <fstream>
#include <string>
#include <cstdlib>
#include <sstream>

std::string ssystem (const char *command) {
    char tmpname [L_tmpnam];
    std::tmpnam ( tmpname );
    std::string scommand = command;
    std::string cmd = scommand + " >> " + tmpname;
    std::system(cmd.c_str());
    std::ifstream file(tmpname, std::ios::in | std::ios::binary );
    std::string result;
    if (file) {
        while (!file.eof()) result.push_back(file.get())
            ;
        file.close();
    }
    remove(tmpname);
    return result;
}

// For Cygwin

int main(int argc, char *argv[])
{
    std::string bash = "FILETWO=/cygdrive/c/*\nfor f in $FILETWO\ndo\necho \"$f\"\ndone ";
    std::string in;
    std::string s = ssystem(bash.c_str());
    std::istringstream iss(s);
    std::string line;
    while (std::getline(iss, line))
    {
        std::cout << "LINE-> " + line + "  length: " << line.length() << std::endl;
    }
    std::cin >> in;
    return 0;
}

4
ฉันได้รับคำเตือนนี้ด้วย gcc: "คำเตือน: การใช้งานtmpnamเป็นสิ่งอันตราย, ใช้งานได้ดีกว่าmkstemp"
Mark Lakata

4

สมมติว่า POSIX, รหัสง่าย ๆ ในการจับ stdout:

#include <sys/wait.h>
#include <unistd.h>
#include <string>
#include <vector>

std::string qx(const std::vector<std::string>& args) {
  int stdout_fds[2];
  pipe(stdout_fds);

  int stderr_fds[2];
  pipe(stderr_fds);

  const pid_t pid = fork();
  if (!pid) {
    close(stdout_fds[0]);
    dup2(stdout_fds[1], 1);
    close(stdout_fds[1]);

    close(stderr_fds[0]);
    dup2(stderr_fds[1], 2);
    close(stderr_fds[1]);

    std::vector<char*> vc(args.size() + 1, 0);
    for (size_t i = 0; i < args.size(); ++i) {
      vc[i] = const_cast<char*>(args[i].c_str());
    }

    execvp(vc[0], &vc[0]);
    exit(0);
  }

  close(stdout_fds[1]);

  std::string out;
  const int buf_size = 4096;
  char buffer[buf_size];
  do {
    const ssize_t r = read(stdout_fds[0], buffer, buf_size);
    if (r > 0) {
      out.append(buffer, r);
    }
  } while (errno == EAGAIN || errno == EINTR);

  close(stdout_fds[0]);

  close(stderr_fds[1]);
  close(stderr_fds[0]);

  int r, status;
  do {
    r = waitpid(pid, &status, 0);
  } while (r == -1 && errno == EINTR);

  return out;
}

การมีส่วนร่วมของรหัสยินดีต้อนรับสำหรับการทำงานเพิ่มเติม:

https://github.com/ericcurtin/execxx


2

คุณสามารถรับเอาต์พุตหลังจากรันสคริปต์โดยใช้ไพพ์ เราใช้ไพพ์เมื่อเราต้องการผลลัพธ์ของโพรเซสลูก

int my_func() {
    char ch;
    FILE *fpipe;
    FILE *copy_fp;
    FILE *tmp;
    char *command = (char *)"/usr/bin/my_script my_arg";
    copy_fp = fopen("/tmp/output_file_path", "w");
    fpipe = (FILE *)popen(command, "r");
    if (fpipe) {
        while ((ch = fgetc(fpipe)) != EOF) {
            fputc(ch, copy_fp);
        }
    }
    else {
        if (copy_fp) {
            fprintf(copy_fp, "Sorry there was an error opening the file");
        }
    }
    pclose(fpipe);
    fclose(copy_fp);
    return 0;
}

ดังนั้นนี่คือสคริปต์ที่คุณต้องการเรียกใช้ วางไว้ในตัวแปรคำสั่งที่มีอาร์กิวเมนต์ที่สคริปต์ของคุณใช้ (ไม่มีอะไรถ้าไม่มีอาร์กิวเมนต์) และไฟล์ที่คุณต้องการดักจับเอาต์พุตของสคริปต์วางไว้ใน copy_fp

ดังนั้น popen จะเรียกใช้สคริปต์ของคุณและวางเอาท์พุทเป็น fpipe จากนั้นคุณก็สามารถคัดลอกทุกอย่างจากสิ่งนั้นไปยังไฟล์เอาต์พุตของคุณ

ด้วยวิธีนี้คุณสามารถดักจับผลลัพธ์ของกระบวนการลูกได้

และกระบวนการอื่นคือคุณสามารถใส่โดยตรง >โอเปอเรเตอร์ในคำสั่งโดยตรงเท่านั้น ดังนั้นหากเราจะใส่ทุกอย่างไว้ในไฟล์ในขณะที่เรารันคำสั่งคุณจะไม่ต้องคัดลอกอะไรเลย

ในกรณีนี้ไม่จำเป็นต้องใช้ท่อ คุณสามารถใช้เพียงsystemและมันจะเรียกใช้คำสั่งและใส่ผลลัพธ์ในไฟล์นั้น

int my_func(){
    char *command = (char *)"/usr/bin/my_script my_arg > /tmp/my_putput_file";
    system(command);
    printf("everything saved in my_output_file");
    return 0;
}

คุณสามารถอ่านบทช่วยสอน YoLinux: การควบคุมผู้บริหารและผู้ควบคุมกระบวนการสำหรับข้อมูลเพิ่มเติม


1

รับทราบว่าคุณสามารถรับเอาท์พุทได้โดยเปลี่ยนเส้นทางไปที่ไฟล์แล้วอ่านมัน

มันถูกแสดงในเอกสารของ std::system

คุณสามารถรับรหัสทางออกได้โดยโทรWEXITSTATUSมาโคร

    int status = std::system("ls -l >test.txt"); // execute the UNIX command "ls -l >test.txt"
    std::cout << std::ifstream("test.txt").rdbuf();
    std::cout << "Exit code: " << WEXITSTATUS(status) << std::endl;
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.