Из ответа Эрика Маленфанта:
AFAIK, нет способа сделать это в
стандарт C ++. В зависимости от вашего
платформа, ваша реализация
стандартная библиотека может предложить (как
нестандартное расширение) фстрим
конструктор, принимающий файловый дескриптор
в качестве ввода. (Это случай для
libstdc ++, IIRC) или ФАЙЛ *.
На основании приведенных выше наблюдений и моих исследований ниже, есть рабочий код в двух вариантах; один для libstdc ++ и другой для Microsoft Visual C ++.
libstdc ++
Существует нестандартный __gnu_cxx::stdio_filebuf
шаблон класса, который наследует std::basic_streambuf
и имеет следующий конструктор
stdio_filebuf (int __fd, std::ios_base::openmode __mode, size_t __size=static_cast< size_t >(BUFSIZ))
с описанием Этот конструктор связывает буфер файлового потока с открытым дескриптором файла POSIX.
Мы создаем его, передавая дескриптор POSIX (строка 1), а затем передаем его конструктору istream как basic_streambuf (строка 2):
#include <ext/stdio_filebuf.h>
#include <iostream>
#include <fstream>
#include <string>
using namespace std;
int main()
{
ofstream ofs("test.txt");
ofs << "Writing to a basic_ofstream object..." << endl;
ofs.close();
int posix_handle = fileno(::fopen("test.txt", "r"));
__gnu_cxx::stdio_filebuf<char> filebuf(posix_handle, std::ios::in); // 1
istream is(&filebuf); // 2
string line;
getline(is, line);
cout << "line: " << line << std::endl;
return 0;
}
Microsoft Visual C ++
Раньше была нестандартная версия конструктора ifstream, принимающего файловый дескриптор POSIX, но она отсутствует как в текущих документах, так и в коде. Существует еще одна нестандартная версия конструктора ifstream, принимающего FILE *
explicit basic_ifstream(_Filet *_File)
: _Mybase(&_Filebuffer),
_Filebuffer(_File)
{ // construct with specified C stream
}
и это не задокументировано (я даже не смог найти какую-либо старую документацию, где она будет присутствовать). Мы вызываем его (строка 1) с параметром, являющимся результатом вызова _fdopen , чтобы получить C поток FILE * из дескриптора файла POSIX.
#include <cstdio>
#include <iostream>
#include <fstream>
#include <string>
using namespace std;
int main()
{
ofstream ofs("test.txt");
ofs << "Writing to a basic_ofstream object..." << endl;
ofs.close();
int posix_handle = ::_fileno(::fopen("test.txt", "r"));
ifstream ifs(::_fdopen(posix_handle, "r")); // 1
string line;
getline(ifs, line);
ifs.close();
cout << "line: " << line << endl;
return 0;
}