将 boost 写入和读取到 POSIX 管道无 boost.iostream

问题描述

我有一个设置,它使用 boost 的序列化将对象存档到文件中。我想要/需要做同样的事情,但写入 POSIX 管道而不是基本文件

一个简单的数据包:

#include <boost/archive/binary_iarchive.hpp>
#include <boost/archive/binary_oarchive.hpp>
#include <boost/serialization/base_object.hpp>
#include <boost/serialization/export.hpp>
#include "AbstractPacket.hpp"

struct SimplePacket final: public AbstractPacket{
    bool value;

    BooleanPacket() noexcept = default;

    template <typename Archive>
    void serialize(Archive& a,const unsigned int) {
        a& boost::serialization::base_object<AbstractPacket>(*this);
        a& value;
    }

    ~SimplePacket() = default;
};

编写简单的数据包:

#include "SimplePacket.hpp"
BOOST_CLASS_EXPORT_GUID(SimplePacket,"SimplePacket")
//...//
void write(Packet* packet){
    std::ofstream file;
    file.open(_client_pipe,std::fstream::out | std::fstream::binary);
    if (file.is_open()) {
        boost::archive::binary_oarchive oa(file);
        oa << packet;
        file.close();
    }
}

这会将数据包(boost 序列化存档)写入文件。但是,我需要写入 posix 管道。 我有 read 使用 boost.iostream 是否可以从其文件描述符打开文件流到 POSIX 管道。 我还没有找到明确的方法来做到这一点; 但是我想不使用boost.iostream来做到这一点,这样的事情可能吗,如果有的话我该怎么做?

解决方法

Boost Asio + Process 一起也可以做到这一点:

使用 Boost 过程

Live On wandbox

size_t write(Packet* packet) {
    boost::asio::streambuf sb;
    {
        std::ostream                    os(&sb);
        boost::archive::binary_oarchive oa(os);
        oa << packet;
    }

    boost::asio::io_context ctx;
    boost::process::async_pipe p(ctx); // opens a pipe (pair of fds)

    size_t bytes = write(p,sb);       // throws system_error on error

    return bytes;
}

如果您已经有了管道,请为 async_pipe 使用适当的构造函数来传递 fds。

仅使用 Boost Asio

此外,如果您已经有了管道 fd,我建议您根本不要使用 Boost Process:

boost::asio::io_context               ctx;
boost::asio::posix::stream_descriptor s(ctx,2); // STDERR

size_t bytes = write(s,sb); // throws system_error on error
s.release();                 // to avoid close on destruction

特别要注意 release() 可能会节省您挠头的时间。

Live On Wandbox

#include <boost/archive/text_iarchive.hpp>
#include <boost/archive/text_oarchive.hpp>
#include <boost/serialization/base_object.hpp>
#include <boost/serialization/export.hpp>

struct AbstractPacket {
    virtual ~AbstractPacket() = default;
    template <typename Archive> void serialize(Archive&,unsigned) {}
};

BOOST_SERIALIZATION_ASSUME_ABSTRACT(AbstractPacket)

struct BooleanPacket : public AbstractPacket {
    bool value;

    template <typename Archive> void serialize(Archive& a,unsigned) {
        a& boost::serialization::base_object<AbstractPacket>(*this);
        a& value;
    }
};

BOOST_CLASS_EXPORT(BooleanPacket)

#include <boost/asio.hpp>

size_t write(AbstractPacket* packet) {
    boost::asio::streambuf sb;
    {
        std::ostream                    os(&sb);
        boost::archive::text_oarchive oa(os);
        oa << packet;
    }

    boost::asio::io_context               ctx;
    boost::asio::posix::stream_descriptor s(ctx,2); // STDERR

    size_t bytes = write(s,sb); // throws system_error on error
    s.release();                 // to avoid close on destruction

    return bytes;
}

#include <iostream>
int main() {
    auto test   = new BooleanPacket(); // please use smart pointers
    test->value = true;

    std::cout << "Bytes written: " << write(test) << "\n";

    delete test; // please use smart pointers
}

打印标准错误:

22 serialization::archive 18 1 13 BooleanPacket 1 0
0 1 0
1 1

和标准输出:

Bytes written: 62

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...