- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我试图理解 Boost.Asio,目的是潜在地使用条件变量结合 Boost.Asio 来实现信号系统。
我看过其他 StackOverflow 问题 boost asio asynchronously waiting on a condition variable , boost::asio async condition , 和 boost condition variable issue ,但这些问题/答案都没有令人满意地触及我的一个基本问题:Is it true that and/or is there a fundamental reason why, Boost.Asio is not applicable to, or a natural fit with , 条件变量?
我的想法是,条件变量是使用操作系统级同步对象在内部实现的(例如,Windows 上的 boost::thread::condition_variable 使用 Windows 操作系统信号量)。因为,根据我目前的理解,boost::asio::io_service 旨在封装 OS 级同步对象,因此条件变量似乎很适合。
的确,与文件操作和套接字操作不同,在操作系统级别通常没有与信号条件相关联的回调函数(我认为 - 我我不确定)。然而,在 Boost.Asio 中实现这样的回调处理程序似乎很简单,只需要求用户提供一个回调函数,该回调函数将在条件变量发出信号时调用——就像用户必须为其他人提供完成处理程序例程一样boost::asio::io_service 服务。
例如(这只是一个快速的想法,不是一个完整的原型(prototype)——它没有包含足够的参数来处理 notify_one() 和 notify_all(),没有说明服务如何知道何时退出,并且可能有其他明显的遗漏或缺陷):
void condition_handler_function() {}
boost::asio::io_service service;
boost::mutex mut;
boost::condition_variable cond;
// The following class is **made up by me** - would such a class be a good idea?
boost::asio::io_service::condition_service
condserv(service, cond, mut, condition_handler_function);
condserv.async_wait_on_signal();
service.run(); // when condition variable is signaled by notify_one(),
// 'handler_function()' would be called
// ... in some other thread, later:
cond.notify_one(); // This would trigger 'handler_function()'
// in this theoretical code
也许,如果我尝试填写代码片段上方提到的缺失细节,我就会清楚这无法以干净的方式工作。然而,这种努力并非微不足道。
因此,我想在这里发布问题。 Boost.Asio 不支持条件变量是否有充分的理由?
附录
我已经更改了帖子的标题以引用“基于事件的接口(interface)”,因为 Tanner 在下面的回答向我澄清了我所询问的实际上是一个基于事件的接口(interface)(不是真正的条件变量) .
最佳答案
Boost.Asio 是一个用于网络和低级 I/O 编程的 C++ 库。因此,操作系统级别的同步对象(例如条件变量)不在库的范围内,更适合 Boost.Thread。 Boost.Asio 作者经常将 boost::asio::io_service
作为应用程序和操作系统之间的桥梁或链接。虽然这可能过于简化,但它在操作系统的 I/O 服务的上下文中。
由于操作启动和完成之间的时间和空间分离,异步编程已经具有先天的复杂性。 Strands提供了一个相当干净的解决方案来提供处理程序的严格顺序调用,而不需要显式锁定。由于锁定是隐式的和线程安全的,应用程序代码可以使用链而不用担心死锁。另一方面,让 boost::asio::io_service::condition_service
对外部提供的对象执行隐式同步可能会将一个复杂的库变成一个复杂的库。应用程序开发人员可能不清楚同步处理程序的互斥锁以及互斥锁的状态。此外,由于隐式锁定,它还引入了应用程序更容易死锁事件处理循环的能力。
如果需要调用基于事件的处理程序,那么一种相当简单的替代方法是使用相同的方法 Boost.Asio 的 timeout server示例使用:boost::asio::deadline_timer
. deadline_timer
的到期时间可以设置为 posix_time::pos_infin
,导致 async_wait
的处理程序仅在定时器被取消后被调用:
cancel()
可以用作 notify_all()
,其中所有未完成的处理程序都排队等待调用。cancel_one()
可以用作 notify_one()
,其中最多有一个未完成的处理程序排队等待调用。一个简单的例子,忽略错误码处理,如下:
#include <iostream>
#include <boost/asio.hpp>
#include <boost/thread.hpp>
class event
{
public:
explicit
event(boost::asio::io_service& io_service)
: timer_(io_service)
{
// Setting expiration to infinity will cause handlers to
// wait on the timer until cancelled.
timer_.expires_at(boost::posix_time::pos_infin);
}
template <typename WaitHandler>
void async_wait(WaitHandler handler)
{
// bind is used to adapt the user provided handler to the deadline
// timer's wait handler type requirement.
timer_.async_wait(boost::bind(handler));
}
void notify_one() { timer_.cancel_one(); }
void notify_all() { timer_.cancel(); }
private:
boost::asio::deadline_timer timer_;
};
void on_event() { std::cout << "on_event" << std::endl; }
int main()
{
boost::asio::io_service io_service;
event event(io_service);
// Add work to service.
event.async_wait(&on_event);
// Run io_service.
boost::thread thread(boost::bind(&boost::asio::io_service::run,
&io_service));
// Trigger event, causing the on_event handler to run.
event.notify_one();
thread.join();
}
关于c++ - 为什么 Boost.Asio 不支持基于事件的接口(interface)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17005258/
我正在尝试在我的代码库中为我正在编写的游戏服务器更多地使用接口(interface),并了解高级概念以及何时应该使用接口(interface)(我认为)。在我的例子中,我使用它们将我的包相互分离,并使
我有一个名为 Widget 的接口(interface),它在我的整个项目中都在使用。但是,它也用作名为 Widget 的组件的 Prop 。 处理此问题的最佳方法是什么?我应该更改我的 Widget
有一个接口(interface)可以是多个接口(interface)之一 interface a {x:string} interface b {y:string} interface c {z:st
我遇到了一种情况,我需要调用第三方服务来获取一些信息。这些服务对于不同的客户可能会有所不同。我的界面中有一个身份验证功能,如下所示。 interface IServiceProvider { bool
在我的例子中,“RequestHandlerProxy”是一个结构,其字段为接口(interface)“IAdapter”,接口(interface)有可能被调用的方法,该方法的输入为结构“Reque
我有一个接口(interface)Interface1,它已由类A实现,并且设置了一些私有(private)变量值,并且我将类A的对象发送到下一个接受输入作为Interface2的类。那么我怎样才能将
假设我有这样的类和接口(interface)结构: interface IService {} interface IEmailService : IService { Task SendAs
有人知道我在哪里可以找到 XML-RPC 接口(interface)的定义(在 OpenERP 7 中)?我想知道创建或获取对象需要哪些参数和对象属性。每个元素的 XML 示例也将非常有帮助。 最佳答
最近,我一直在阅读有关接口(interface)是抽象的错误概念的文章。一篇这样的帖子是http://blog.ploeh.dk/2010/12/02/InterfacesAreNotAbstract
如果我有一个由第三方实现的现有 IInterface 后代,并且我想添加辅助例程,Delphi 是否提供了任何简单的方法来实现此目的,而无需手动重定向每个接口(interface)方法?也就是说,给定
我正在尝试将 Article 数组分配给我的 Mongoose 文档,但 Typescript 似乎不喜欢这样,我不知道为什么它显示此警告/错误,表明它不可分配. 我的 Mongoose 模式和接口(
我有两个接口(interface): public interface IController { void doSomething(IEntity thing); } public inte
是否可以创建一个扩展 Serializable 接口(interface)的接口(interface)? 如果是,那么扩展接口(interface)的行为是否会像 Serilizable 接口(int
我试图在两个存储之间创建一个中间层,它从存储 A 中获取数据,将其转换为相应类型的存储 B,然后存储它。由于我需要转换大约 50-100 种类型,我希望使用 map[string]func 并根据 s
我正在处理一个要求,其中我收到一个 JSON 对象,其中包含一个日期值作为字符串。我的任务是将 Date 对象存储在数据库中。 这种东西: {"start_date": "2019-05-29", "
我们的方法的目标是为我们现有的 DAO 和模型类引入接口(interface)。模型类由各种类型的资源 ID 标识,资源 ID 不仅仅是随机数,还带有语义和行为。因此,我们必须用对象而不是原始类型来表
Collection 接口(interface)有多个方法。 List 接口(interface)扩展了 Collection 接口(interface)。它声明与 Collection 接口(int
我有一个 Java 服务器应用程序,它使用 Jackson 使用反射 API 对 DTO 进行一般序列化。例如对于这个 DTO 接口(interface): package com.acme.libr
如果我在 Kotlin 中有一个接口(interface): interface KotlinInterface { val id: String } 我可以这样实现: class MyCla
我知道Java中所有访问修饰符之间的区别。然而,有人问了我一个非常有趣的问题,我很难找到答案:Java 中的 private 接口(interface)和 public 接口(interface)有什
我是一名优秀的程序员,十分优秀!