如何检测boost tcp套接字何时断开连接

时间:2013-11-10 08:03:22

标签: c++ sockets boost tcp boost-asio

假设我有一个套接字:

std::shared_ptr<tcp::socket> socket( new tcp::socket(acceptor.get_io_service()) );
acceptor.async_accept( *socket, std::bind( handleAccept, this, std::placeholders::_1, socket, std::ref(acceptor)) );

我将weak_ptr存储到容器中的所述套接字中。我需要这个,因为我想允许客户请求其他客户的列表,这样他们就可以互相发送消息。

clients_.insert(socket);  // pseudocode

然后我运行一些异步操作

socket->async_receive( boost::asio::buffer(&(*header), sizeof(Header))
    , 0
    , std::bind(handleReceiveHeader, this, std::placeholders::_1, std::placeholders::_2, header, socket));

如何检测连接何时关闭,以便从容器中取出插座?

clients_.erase(socket); // pseudocode

2 个答案:

答案 0 :(得分:19)

TCP套接字断开连接通常由asioeofconnection_reset中发出信号。 E.g。

  void async_receive(boost::system::error_code const& error,
                     size_t bytes_transferred)
  {
    if ((boost::asio::error::eof == error) ||
        (boost::asio::error::connection_reset == error))
    {
      // handle the disconnect.
    }
    else
    {
       // read the data 
    }
  }

我使用boost::signals2来表示断开连接,尽管您总是可以将指向函数的指针传递给套接字类,然后调用它。

请注意您的套接字和回调生命周期,请参阅:boost-async-functions-and-shared-ptrs

答案 1 :(得分:1)

有很多选择,其中一些是:

  1. 当您将weak_ptr存储在容器中时 - 它不会延长套接字的生命周期,因此当您的处理程序获得boost::asio::error::eof(或其他)时,它将不会复制/移动{ {1}},套接字将被删除(如果你没有任何其他shared_ptr)。因此,您可以执行以下操作:shared_ptr

  2. 检查处理程序中的错误代码 - 它将指示何时关闭连接。使用此信息 - 您可以从处理程序本身调用if(socket.expired()) clients_.erase(socket);

  3.   

    你能举一个#2的例子吗?

    这将是:

    clients_.erase