4

我正在调查 c++ 库以实现可移植、阻止对文件系统和网络的 I/O 访问。看起来,并且将在他们三个boost::filesystem之间完成这项工作。boost::iostreamsboost::asio

需要明确的是,我目前对 ; 的异步方面不感兴趣boost::asio。我只想要一个可移植的、阻塞的网络接口。

深入研究,我看到boost::iostreams了设备的概念,每个设备都有一个相关的模式概念。双向模式似乎专门为流式访问全双工 TCP 连接量身定制。惊人的。

boost::iostreams似乎不支持实际打开 TCP 连接(与本地文件系统不同)。这很好,肯定boost::asio会让我打开连接,将其适当地建模为双向Device,并将其包装在boost::iostreams::stream.

..除非它不会?我明白boost::asio::ip::tcp::iostream了,这将取代boost::iostreams::stream,但大概不会充当Device.

我理解它的tcp::iostream行为类似,但我仍然更喜欢只针对一个界面而不是两个界面来学习和编码。具体来说,处理两种错误处理机制和异常层次结构并不是很容易接受。

那么问题来了:我是瞎了吗?也许两个库之间存在一个适配器,我错过了谷歌搜索。或者也许有人已经发布了这样的适配器作为我可以加入的第 3 方组件?

4

1 回答 1

7

我不知道直接映射。但是,如果您有兴趣,编写这样的设备相当简单。此版本会抛出boost::system::system_error非 EOF 错误,但您可以选择执行其他操作。

#include <iosfwd>

#include <boost/asio/io_service.hpp>
#include <boost/asio/ip/tcp.hpp>
#include <boost/asio/buffer.hpp>
#include <boost/iostreams/categories.hpp>
#include <boost/system/system_error.hpp>


class asio_stream_device
{
public:
    typedef char char_type;
    typedef boost::iostreams::bidirectional_device_tag category;

    explicit asio_stream_device(boost::asio::ip::tcp::socket& sock) : socket_(sock)
    {

    }

    std::streamsize read(char* s, std::streamsize n)
    {
        // Read up to n characters from the underlying data source
        // into the buffer s, returning the number of characters
        // read; return -1 to indicate EOF
        boost::system::error_code ec;

        std::size_t rval = socket_.read_some(boost::asio::buffer(s, n), ec);
        if (!ec)
        {
            return rval;
        }
        else if (ec == boost::asio::error::eof)
        {
            return -1;
        }
        else
        {
            throw boost::system::system_error(ec,"read_some");
        }

    }


    std::streamsize write(const char* s, std::streamsize n)
    {
        // Write up to n characters to the underlying
        // data sink into the buffer s, returning the
        // number of characters written

        boost::system::error_code ec;
        std::size_t rval = socket_.write_some(boost::asio::buffer(s, n), ec);
        if (!ec)
        {
            return rval;
        }
        else if (ec == boost::asio::error::eof)
        {
            return -1;
        }
        else
        {
            throw boost::system::system_error(ec,"write_some");
        }

    }



private:

    boost::asio::ip::tcp::socket& socket_;

};

基本上,正常打开/连接套接字,然后将其传递给构造函数。该示例只是读取并输出到屏幕。

void test
{
   namespace asio = boost::asio;
   namespace io = boost::iostreams;

   asio::io_service service;
   asio::ip::tcp::socket socket(service);


   asio::ip::tcp::endpoint remote -  ...; ////

   socket.connect(remote);

   io::stream<asio_stream_device> str(socket);

   std::string line;

   while (std::getline(str, line)) {
    std::cout << line << std::endl;
   }
}
于 2012-08-19T01:41:57.760 回答