- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我经常在代码中看到这种模式,将 shared_from_this
作为第一个参数绑定(bind)到成员函数并使用 async_*
函数调度结果。这是另一个问题的示例:
void Connection::Receive()
{
boost::asio::async_read(socket_,boost::asio::buffer(this->read_buffer_),
boost::bind(&Connection::handle_Receive,
shared_from_this(),
boost::asio::placeholders::error,
boost::asio::placeholders::bytes_transferred));
}
使用 shared_from_this()
而不是 this
的唯一原因是在调用成员函数之前保持对象处于事件状态。但除非在某处有某种 boost 魔法,因为 this
指针的类型是 Connection*
,这就是 handle_Receive
可以采用的所有内容,而 smart返回的指针应立即转换为常规指针。如果发生这种情况,则没有任何东西可以使对象保持活力。当然,调用 shared_from_this
时没有指针。
但是,我经常看到这种模式,我不敢相信它会像我认为的那样完全被打破。当操作完成时,是否有一些 Boost 魔术导致 shared_ptr 稍后转换为常规指针?如果是这样,这是否记录在某处?
特别是,它是否在某处记录了共享指针将保持存在直到操作完成?在强指针上调用 get_pointer
然后在返回的指针上调用成员函数是不够的,除非强指针直到成员函数返回才被销毁。
最佳答案
简而言之,boost::bind
创建 boost::shared_ptr<Connection>
的拷贝从 shared_from_this()
返回, 和 boost::asio
可以创建处理程序的拷贝。处理程序的拷贝将保持事件状态,直到发生以下情况之一:
run()
所在的线程调用。 , run_one()
, poll()
或 poll_one()
成员函数已被调用。io_service
被销毁了。io_service::service
拥有处理程序的通过 shutdown_service()
关闭.以下是文档的相关摘录:
boost::bind documentation :
The arguments that
bind
takes are copied and held internally by the returned function object.
boost::asio io_service::post
:
The
io_service
guarantees that the handler will only be called in a thread in which therun()
,run_one()
,poll()
orpoll_one()
member functions is currently being invoked. [...] Theio_service
will make a copy of the handler object as required.
boost::asio io_service::~io_service
:
Uninvoked handler objects that were scheduled for deferred invocation on the
io_service
, or any associated strand, are destroyed.
Where an object's lifetime is tied to the lifetime of a connection (or some other sequence of asynchronous operations), ashared_ptr
to the object would be bound into the handlers for all asynchronous operations associated with it. [...] When a single connection ends, all associated asynchronous operations complete. The corresponding handler objects are destroyed, and allshared_ptr
references to the objects are destroyed.
虽然日期为 (2007),但 Networking Library Proposal for TR2 (Revision 1)源自 Boost.Asio。栏目5.3.2.7. Requirements on asynchronous operations
为 async_
的参数提供了一些详细信息功能:
In this clause, an asynchronous operation is initiated by a function that is named with the prefix
async_
. These functions shall be known as initiating functions. [...] The library implementation may make copies of the handler argument, and the original handler argument and all copies are interchangeable.The lifetime of arguments to initiating functions shall be treated as follows:
- If the parameter is declared as a const reference or by-value [...] the implementation may make copies of the argument, and all copies shall be destroyed no later than immediately after invocation of the handler.
[...] Any calls made by the library implementation to functions associated with the initiating function's arguments will be performed such that calls occur in a sequence call1 to calln, where for all i, 1 ≤ i < n, calli precedes call i+1.
因此:
shared_ptr<Connection>
的拷贝,增加 Connection
的引用计数实例,而 handler 的拷贝仍然存在。io_serive::service
时未完成,则会发生这种情况。正在关闭或 io_service
被摧毁。在示例中,处理程序的拷贝将被销毁,从而减少 Connection
的引用计数。 ,并可能导致 Connection
要销毁的实例。Connection
的引用计数。 ,并可能导致其被破坏。asnyc_
相关的函数的参数,将按顺序执行,而不是并发执行。这包括 io_handler_deallocate
和 io_handler_invoke
.这保证了在调用 handler 时不会释放 handler。在 boost::asio
的大部分地区实现时,处理程序被复制或移动到堆栈变量,一旦执行退出声明它的 block ,就允许销毁发生。在示例中,这确保了 Connection
的引用计数在调用 handler 期间将至少为 1。关于c++ - boost async_* 函数和 shared_ptr 的,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11356742/
我正在尝试使用boost.spirit的qi库解析某些内容,而我遇到了一个问题。根据spirit docs,a >> b应该产生类型为tuple的东西。但这是boost::tuple(又名 fusio
似乎有/正在努力做到这一点,但到目前为止我看到的大多数资源要么已经过时(带有死链接),要么几乎没有信息来实际构建一个小的工作样本(例如,依赖于boost program_options 以构建可执行文
我对 Boost.Log 的状态有点困惑。这是 Boost 的官方部分,还是尚未被接受?当我用谷歌搜索时,我看到一些帖子谈论它在 2010 年是如何被接受的,等等,但是当我查看最后一个 Boost 库
Boost 提供了两种不同的实现 string_view ,这将成为 C++17 的一部分: boost::string_ref在 utility/string_ref.hpp boost::stri
最近,我被一家GIS公司雇用来重写他们的旧地理信息库。所以我目前正在寻找一个好的计算几何库。我看过CGAL,这真是了不起,但是我的老板想要免费的东西。 所以我现在正在检查Boost.Geometry。
假设我有一个无向图 G。假设我添加以下内容 add_edge(1,2,G); add_edge(1,3,G); add_edge(0,2,G); 现在我再说一遍: add_edge(0,2,G); 我
我使用 CMake 来查找 Boost。找到了 Boost,但 CMake 出错了 Imported targets not available for Boost version 请参阅下面的完整错
我是 boost::fusion 和 boost::mpl 库的新手。谁能告诉我这两个库之间的主要区别? 到目前为止,我只使用 fusion::vector 和其他一些简单的东西。现在我想使用 fus
这个问题已经有答案了: 已关闭10 年前。 Possible Duplicate: What are the benefits of using Boost.Phoenix? 所以我开始阅读 boos
我正在尝试获得一个使用 Boost.Timer 的简单示例,用于一些秒表性能测量,但我不明白为什么我无法成功地将 Boost.Timer 链接到 Boost.Chrono。我使用以下简单脚本从源代码构
我有这样的东西: enum EFood{ eMeat, eFruit }; class Food{ }; class Meat: public Food{ void someM
有人可以告诉我,我如何获得boost::Variant处理无序地图? typedef boost::variant lut_value;unordered_map table; 我认为有一个用于boo
我对 Boost.Geometry 中的环和多边形感到困惑。 在文档中,没有图形显示什么是环,什么是多边形。 谁能画图解释两个概念的区别? 最佳答案 在 Boost.Geometry 中,多边形被定义
我正在使用 boost.pool,但我不知道何时使用 boost::pool<>::malloc和 boost::pool<>::ordered_malloc ? 所以, boost::pool<>:
我正在尝试通过 *boost::fast_pool_allocator* 使用 *boost::container::flat_set*。但是,我收到编译错误。非常感谢您的意见和建议。为了突出这个问题
sau_timer::sau_timer(int secs, timerparam f) : strnd(io), t(io, boost::posix_time::seconds(secs)
我无法理解此功能的文档,我已多次看到以下内容 tie (ei,ei_end) = out_edges(*(vi+a),g); **g**::out_edge_iterator ei, ei_end;
我想在 C++ 中序列化分层数据结构。我正在处理的项目使用 boost,所以我使用 boost::property_tree::ptree 作为我的数据节点结构。 我们有像 Person 这样的高级结
我需要一些帮助来解决这个异常,我正在实现一个 NPAPI 插件,以便能够使用来自浏览器扩展的本地套接字,为此我正在使用 Firebreath 框架。 对于套接字和连接,我使用带有异步调用的 Boost
我尝试将 boost::bind 与 boost::factory 结合使用但没有成功 我有这个类 Zambas 有 4 个参数(2 个字符串和 2 个整数)和 class Zambas { publ
我是一名优秀的程序员,十分优秀!