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> [关闭]