Чтение popen приводит к С++
Я пишу приложение на С++, и мне нужно прочитать результат системной команды.
Я использую popen()
более или менее, как показано ниже:
const int MAX_BUFFER = 2048;
string cmd="ls -l";
char buffer[MAX_BUFFER];
FILE *stream = popen(cmd.c_str(), "r");
if (stream){
while (!feof(stream))
{
if (fgets(buffer, MAX_BUFFER, stream) != NULL)
{
//here is all my code
}
}
pclose(stream);
}
Я пытался переписать это по-другому. Я видел несколько нестандартных решений, таких как:
FILE *myfile;
std::fstream fileStream(myfile);
std::string mystring;
while(std::getline(myfile,mystring))
{
// .... Here I do what I need
}
Мой компилятор этого не принимает.
Как я могу читать из popen
в С++?
Ответы
Ответ 1
Ваш пример:
FILE *myfile;
std::fstream fileStream(myfile);
std::string mystring;
while(std::getline(myfile,mystring))
Не работает, потому что, хотя вы очень близки, стандартная библиотека не предоставляет fstream
, которая может быть построена из FILE*
. Boost iostreams, однако, предоставляет iostream
, который может быть создан из файлового дескриптора, и вы можете получить его из FILE*
, вызывая fileno
.
например:.
typedef boost::iostreams::stream<boost::iostreams::file_descriptor_sink>
boost_stream;
FILE *myfile;
// make sure to popen and it succeeds
boost_stream stream(fileno(myfile));
stream.set_auto_close(false); // https://svn.boost.org/trac/boost/ticket/3517
std::string mystring;
while(std::getline(stream,mystring))
Не забывайте еще pclose
.
Примечание. Более новые версии boost не рекомендовали конструктор, который занимает всего fd
. Вместо этого вам нужно передать один из boost::iostreams::never_close_handle
или boost::iostreams::close_handle
в качестве обязательного второго аргумента конструктору.
Ответ 2
Вот что я написал давно, может вам помочь. Это может иметь некоторые ошибки.
#include <vector>
#include <string>
#include <stdio.h>
#include <iostream>
bool my_popen (const std::string& cmd,std::vector<std::string>& out ) {
bool ret_boolValue = true;
FILE* fp;
const int SIZEBUF = 1234;
char buf [SIZEBUF];
out = std::vector<std::string> ();
if ((fp = popen(cmd.c_str (), "r")) == NULL) {
return false;
}
std::string cur_string = "";
while (fgets(buf, sizeof (buf), fp)) {
cur_string += buf;
}
out.push_back (cur_string.substr (0, cur_string.size () - 1));
pclose(fp);
return true;
}
int main ( int argc, char **argv) {
std::vector<std::string> output;
my_popen("ls -l > /dev/null ", output);
for ( std::vector<std::string>::iterator itr = output.begin();
itr != output.end();
++itr) {
std::cout << *itr << std::endl;
}
}