boost::asio - 知道何时必须关闭/关闭连接

Posted

技术标签:

【中文标题】boost::asio - 知道何时必须关闭/关闭连接【英文标题】:boost::asio - know when the conection has to be shutdown/closed 【发布时间】:2012-02-03 15:01:50 【问题描述】:

我实现了一个协议(袜子),该协议要求我的服务器将来自客户端的连接中继到目的地。

我实现中继部分的方式是使用这样的:

socket_.async_read_some(boost::asio::buffer(dataClient_, 1024),
boost::bind(&ProxySocksSession::HandleClientProxyRead, this,
 boost::asio::placeholders::error,
 boost::asio::placeholders::bytes_transferred));

remoteSock_.async_read_some(boost::asio::buffer(dataRemote_, 1024),
boost::bind(&ProxySocksSession::HandleRemoteProxyRead, this,
 boost::asio::placeholders::error,
 boost::asio::placeholders::bytes_transferred));

当然还有更多代码 - 那里有处理程序将来自 socket_ 的数据中继并将其发送到 remoteSock_,反之亦然(来自 remoteSock_ 的所有数据都中继到 socket_)

我看到了异步 tcp 服务器回显示例 (http://www.boost.org/doc/libs/1_38_0/doc/html/boost_asio/example/echo/async_tcp_echo_server.cpp),其中关闭连接的逻辑只是删除​​连接对象 - 删除它 - (关闭它拥有的通信套接字)when 在处理程序中收到了 boost::system::error_code。

我应该如何自己处理这个案子?这次我有 2 个套接字上的数据,如果我不干净地关闭,我可能会在所有数据传输之前关闭(例如来自客户端的套接字 - socket_ - 可能会关闭连接但是 - remoteSock - 可能仍在尝试发送数据)。

编辑我已经更新了我的代码,如果我检测到其中一个套接字(remoteSock_ 或 socket_)读/写处理程序报告了 boost::system::error_code,我会执行以下操作为了关闭通信:

void ProxySocksSession::Shutdown()

    if (!shutdownInProgress_)
    
        std::cout << "Shuting down ..." << std::endl;
        shutdownInProgress_ = true;
        remoteSock_.shutdown((boost::asio::ip::tcp::socket::shutdown_both));
        remoteSock_.close();
        socket_.shutdown((boost::asio::ip::tcp::socket::shutdown_both));
        socket_.close();
        parentConnection_.Shutdown();
    

问题是,即使我在套接字上调用 shutdown() 和 close(),我仍然会收到对套接字处理程序的调用(它们在同一个类中,ProxySocksSession)。当这些来临时,我的 ProxySocksSession 实例已经被删除(删除由上面的 parentConnection_.Shutdown() 完成)

【问题讨论】:

async_xxxxx 函数具有完成回调。当操作仍处于挂起状态时,由您决定是否关闭连接。 @Dave 是的,但在某些时候我仍然必须销毁类“ProxySocksSession”以释放内存资源。问题是我在“父”类中将 ProxySocksSession 作为指针/智能指针,当我确定通信完成时,我告诉父类是时候销毁自身(并随之销毁 ProxySocksSession 成员) 有一个很好的关于对象生命周期(和管理它们)的视频教程,由 asio creator 制作:Thinking Asynchronously: Designing Applications with Boost Asio 【参考方案1】:

我设法提出了一个可行的解决方案(不会导致所描述的问题)。 我还在下面包含处理函数的骨架,以便了解这个想法:

void ProxySocksSession::Start()

    socket_.async_read_some(boost::asio::buffer(dataClient_, 1024),
       boost::bind(&ProxySocksSession::HandleClientProxyRead, shared_from_this(),
           boost::asio::placeholders::error,
           boost::asio::placeholders::bytes_transferred));

    remoteSock_.async_read_some(boost::asio::buffer(dataRemote_, 1024),
           boost::bind(&ProxySocksSession::HandleRemoteProxyRead, shared_from_this(),
           boost::asio::placeholders::error,
           boost::asio::placeholders::bytes_transferred));


// received data from socks5 client - completion handler
void ProxySocksSession::HandleClientProxyRead(const boost::system::error_code& error,
      size_t bytes_transferred)

     if (!error && !this->shutdownInProgress_)
     
         // relay data coming from client to remote endpoint -> async write to remoteSock_
         // async read some more data from socket_
      
      else
      
         Shutdown();
      
 

//received data from socks5 remote endpoint (socks5 client destination)
void ProxySocksSession::HandleRemoteProxyRead(const boost::system::error_code& error,
      size_t bytes_transferred)

     if (!error && !this->shutdownInProgress_)
     
         // relay data coming from remote endpoint to client -> async write to socket__
         // async read some more data from remoteSock_
      
      else
      
         Shutdown();
      
 

void ProxySocksSession::Shutdown()

    if (!shutdownInProgress_)
    
        std::cout << "Shuting down ..." << std::endl;
        shutdownInProgress_ = true;
        //remoteSock_.close(); -- no need as it is closed automatically as part of parentConnection_ shutdown/deletion
        //socket_.close(); -- no need as it is closed automatically as part of parentConnection_ shutdown/deletion
        parentConnection_.Shutdown();
    

key 是我在处理完成处理程序时使用了 shared_from_this()。通过这种方式,我确保 删除 ProxySocksSession 实例没有由 parentConnection_ 实例完成,该实例在 ProxySocksSession 之前上有一个 shared_ptr > 所有 ProxySocksSession 处理程序在套接字关闭时被调用。

【讨论】:

有关 shared_from_this() 为何/如何工作的详细信息,与 boost::asio 相关,请查看boost.org/doc/libs/1_45_0/doc/html/boost_asio/tutorial/…。 - “我们将使用 shared_ptr 和 enable_shared_from_this 因为我们希望 tcp_connection 对象只要有引用它的操作就保持活动状态” 感谢您提供的 boost 文档链接,这正是我想要的

以上是关于boost::asio - 知道何时必须关闭/关闭连接的主要内容,如果未能解决你的问题,请参考以下文章

boost :: asio取消或关闭对async_handle不起作用

“xtime: ambiguous symbol”错误,包括 <boost/asio.hpp> [关闭]

boost::asio 无法干净地关闭 TCP 连接

从浏览器中检索我在 cpp 中的代理的地址(boost::Asio)[关闭]

如何优雅地关闭 boost asio ssl 客户端?

Boost asio 获取“错误代码:125 操作已取消”[关闭]