1. ホーム
  2. c++

[解決済み] POSIXファイルディスクリプタからc++のfstreamを構築する方法とは?

2022-11-20 10:24:13

質問

私は基本的に fdopen() の C++ 版を探しています。 私はこれについて少し研究し、それは簡単であるべきであるように見えるが、非常に複雑であることが判明するものの1つです。 私はこの信念の中で何かを見逃しているのでしょうか(つまり、本当に簡単なのでしょうか)? そうでない場合、これを処理するための良いライブラリがどこかにあるのでしょうか?

EDIT: 私の例の解決策を別の答えに移動しました。

どのように解決するのですか?

Éric Malenfant氏の回答より。

AFAIKでは、これを行う方法はありません。 標準のC++にはありません。プラットフォームによっては プラットフォームによっては、標準ライブラリの実装が 標準ライブラリの実装では、(非標準の拡張として) 標準外の拡張として)fstream を入力とするコンストラクタを提供するかもしれません。 を入力とする (これは libstdc++ の場合です)または FILE* です。

上記の観察と以下の私の調査に基づいて、2つのバリエーションで動作するコードがあります; 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++

以前は、非標準の バージョン には、POSIX ファイルディスクリプタを受け取る ifstream のコンストラクタがありましたが、これは 現在の ドキュメントにもコードにもありません。FILE*を受け取るifstreamのコンストラクタの別の非標準バージョンがあります。

explicit basic_ifstream(_Filet *_File)
    : _Mybase(&_Filebuffer),
        _Filebuffer(_File)
    {   // construct with specified C stream
    }

となっていて、ドキュメント化されていません(それが存在するような古いドキュメントも見つけられませんでした)。を呼び出した結果をパラメータとして、それを呼び出しています (1行目)。 _fdopen を呼び出して、POSIX ファイル ハンドルから C ストリーム FILE* を取得した結果をパラメーターとして呼び出します。

#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;
}