C++ 异步编程:Boost.Asio
Boost.Asio 是一个用于网络和低级 I/O 编程的跨平台 C++ 库,它使用现代 C++ 方法为开发人员提供一致的异步模型
一个异步使用计时器的样例
#include <iostream>
#include <boost/asio.hpp>
void print(const boost::system::error_code & /* e */)
{
std::cout <<“hello world!” << std::endl;
}
int main()
{
boost :: asio :: io_context me; // 提供对 i/o 功能的访问
boost :: asio :: steady_timer t(io,boost :: asio :: chrono :: seconds(5));
t.async_wait(print); // 插入回调函数
io.run();
return;
}
asio 库提供了一种保证,即只能从当前调用 io_context::run() 的线程调用回调处理程序
io_context::run() 函数将继续运行,它的工作是计时器上的异步等待,在计时器到期并且回调完成之前调用不会返回
在调用 io_context::run() 前需要给 io_context 设定一些工作,如果省略了 t.async_wait(print); 的调用,io_context::run() 将立刻返回
多次触发计时器
要实现重复计时器,需要在回调函数中更改计时器的到期时间,然后启动新的异步等待
设定计时器将在第 6 次触发时停止程序
#include <iostream>
#include <boost/asio.hpp>
#include <boost/bind.hpp>
// 增加指向计时器对象的指针以访问计时器对象,以及一个计数器 count
void print(const boost::system::error_code & /*e*/,
boost::asio::steady_timer * t, int * count)
{
// 这里没有明确的调用要求 io_context 停止,而是通过 count == 5 时不再对计时器启动新的异步等待,io_context 将结束工作并停止运行
if (*count < 5)
{
std::cout << *count << std::endl;
++(*count);
// 将计时器到期时间向后延时 1 秒
t->expires_at(t->expiry() + boost::asio::chrono::seconds(1));
// 启动一个新的异步等待,使用 boost::bind 使需要指定与回调函数参数列表相匹配的参数
t->async_wait(boost::bind(print,
boost::asio::placeholders::error, t, count));
}
}
int main()
{
boost::asio::io_context io;
int count = 0;
boost::asio::steady_timer t(io, boost::asio::chrono::seconds(1));
t.async_wait(boost::bind(print,
boost::asio::placeholders::error, &t, &count));
io.run();
std::cout << "Final count is " << count << std::endl;
return 0;
}
使用类的成员函数作回调处理
#include <iostream>
#include <boost/asio.hpp>
#include <boost/bind.hpp>
class printer
{
public:
// 构造时引用 io_context 对象,使用它初始化 timer
printer(boost::asio::io_context& io)
: timer_(io, boost::asio::chrono::seconds(1)),
count_(0)
{
// 使用 bind 绑定当前对象的 this 指针,利用成员 count 控制计时器的执行
timer_.async_wait(boost::bind(&printer::print, this));
}
// 在析构中打印结果
~printer()
{
std::cout << "Final count is " << count_ << std::endl;
}
// 作为类的成员函数,无需再传入参数,直接使用当前对象的成员变量
void print()
{
if (count_ < 5)
{
std::cout << count_ << std::endl;
++count_;
timer_.expires_at(timer_.expiry() + boost::asio::chrono::seconds(1));
timer_.async_wait(boost::bind(&printer::print, this));
}
}
private:
boost::asio::steady_timer timer_;
int count_;
};
int main()
{
// main 里的调用简单了很多
boost::asio::io_context io;
printer p(io);
io.run();
return 0;
}
多线程中的回调函数同步
asio 库提供了一种保证,即只能从当前调用 io_context::run() 的线程调用回调函数。因此,仅从一个线程调用 io_context::run() 回调函数无法并发运行
这里需要使用 io_context::strand 来控制分配 handler 的执行,它可以保证无论 io_context::run() 的线程有多少,同时只能有一个 handler 程序执行,保证了共享资源的线程安全
在本例中 handler (print1 / print2) 所访问的共享资源包括 std::cout 和成员变量 count_
#include <iostream>
#include <boost/asio.hpp>
#include <boost/thread/thread.hpp>
#include <boost/bind.hpp>
class printer
{
public:
printer(boost::asio::io_context& io)
: strand_(io), // strand 成员,用于控制 handler 的执行
timer1_(io, boost::asio::chrono::seconds(1)), // 运行两个计时器
timer2_(io, boost::asio::chrono::seconds(1)),
count_(0)
{
// 启动异步操作时,每个 handler 都绑定到 strand 对象
// bind_executor() 返回一个新的 handler,它将自动调度其包含的 printer::print1
// 通过将 handler 绑定到同一个 strand,保证它不会同时执行
timer1_.async_wait(boost::asio::bind_executor(strand_,
boost::bind(&printer::print1, this)));
timer2_.async_wait(boost::asio::bind_executor(strand_,
boost::bind(&printer::print2, this)));
}
~printer()
{
std::cout << "Final count is " << count_ << std::endl;
}
void print1()
{
if (count_ < 10)
{
std::cout << "Timer 1: " << count_ << std::endl;
++count_;
timer1_.expires_at(timer1_.expiry() + boost::asio::chrono::seconds(1));
timer1_.async_wait(boost::asio::bind_executor(strand_,
boost::bind(&printer::print1, this)));
}
}
void print2()
{
if (count_ < 10)
{
std::cout << "Timer 2: " << count_ << std::endl;
++count_;
timer2_.expires_at(timer2_.expiry() + boost::asio::chrono::seconds(1));
timer2_.async_wait(boost::asio::bind_executor(strand_,
boost::bind(&printer::print2, this)));
}
}
private:
boost::asio::io_context::strand strand_;
boost::asio::steady_timer timer1_;
boost::asio::steady_timer timer2_;
int count_;
};
int main()
{
boost::asio::io_context io;
printer p(io);
// main 函数现在有两个线程调用 io_context::run(),主线程和由 boost::thread 对象执行的额外线程
boost::thread t(boost::bind(&boost::asio::io_context::run, &io));
io.run();
t.join();
return 0;
}
参考:
How strands work and why you should use them
C++ 异步编程:Boost.Asio的更多相关文章
- Boost.Asio c++ 网络编程翻译(21)
同步VS异步 Boost.Asio的作者做了一个非常惊艳的工作:它能够让你在同步和异步中自由选择,从而更好的适应你的应用. 在之前的章节中,我们学习了每种类型应用的框架,比方同步client,同步服务 ...
- boost::asio译文
Christopher Kohlhoff Copyright © 2003-2012 Christopher M. Kohlhoff 以Boost1.0的软件授权进行发布(见附带的LICENS ...
- Boost.Asio技术文档
Christopher Kohlhoff Copyright © 2003-2012 Christopher M. Kohlhoff 以Boost1.0的软件授权进行发布(见附带的LICENSE_1_ ...
- Boost.Asio的使用技巧
基本概念 Asio proactor I/O服务 work类 run() vs poll() stop() post() vs dispatch() buffer类 缓冲区管理 I/O对象 socke ...
- boost::asio学习(定时器)
#include <boost/asio.hpp> #include <iostream> void handle1(const boost::system::error_co ...
- boost.asio系列(一)——deadline_timer
一.构造函数 一个deadline_timer只维护一个超时时间,一个deadline_timer不同时维护多个定时器.在构造deadline_timer时指定时间: basic_deadline_t ...
- Boost.Asio基础(五) 异步编程初探
异步编程 本节深入讨论异步编程将遇到的若干问题.建议多次阅读,以便吃透这一节的内容,这一节是对整个boost.asio来说是非常重要的. 为什么须要异步 如前所述,通常同步编程要比异步编程更简单.同步 ...
- Boost.Asio c++ 网络编程翻译(20)
异步服务端 这个图表是相当复杂的:从Boost.Asio出来你能够看到4个箭头指向on_accept.on_read,on_write和on_check_ping. 着也就意味着你永远不知道哪个异步调 ...
- boost::asio 的同、异步方式
转自:http://blog.csdn.net/zhuky/archive/2010/03/10/5364574.aspx Boost.Asio是一个跨平台的网络及底层IO的C++编程库,它使用现代C ...
随机推荐
- 第2课第2节_Java面向对象编程_封装性_P【学习笔记】
摘要:韦东山android视频学习笔记 面向对象程序的三大特性之封装性:把属性和方法封装在一个整体,同时添加权限访问. 1.封装性的简单程序如下,看一下第19行,如果我们不对age变量进行权限的管控 ...
- MXNet/Gluon 中网络和参数的存取方式
https://blog.csdn.net/caroline_wendy/article/details/80494120 Gluon是MXNet的高层封装,网络设计简单易用,与Keras类似.随着深 ...
- mysql 日期总结
select to_days(now()) #737733 select date_format(now(),'%Y-%m-%d') #2019-11-05 select CURRENT_DATE() ...
- yii2 下的redis常用命令集合
<?php \Yii::$app->redis->set('user','aaa'); \Yii::$app->redis->set('user2','bbb'); \Y ...
- [原][OE][官方例子]osgearth_annotation OE地球添加热点标签
OE所有官方例子 OE代码样例 /* -*-c++-*- */ /* osgEarth - Dynamic map generation toolkit for OpenSceneGraph * Co ...
- Dart运算符条件判断类型转换
/* 1.Dart运算符: 算术运算符 + - * / ~/ (取整) %(取余) 关系运算符 == != > < >= <= 逻辑运算符 ! && || 赋值 ...
- 使用poi解决导出excel内下拉框枚举项较多的问题
废话少说,直接上代码: package com.fst.attachment.controller; import java.io.FileOutputStream; import org.apach ...
- Oracle中复制表的方法(create as select、insert into select、select into)
转: Oracle中复制表的方法(create as select.insert into select.select into) 2018-07-30 22:10:37 小白白白又白cdllp 阅读 ...
- 单独使用MyBatis的简单示例
单独使用MyBatis的简单示例:mybaties-config.xml:MyBatis配置文件 <?xml version="1.0" encoding="UTF ...
- elasticsearch关键词查询不分词
$query = [ 'bool' => [ 'must' => [ 'match_phrase' => ['content' => $word] //$word词不被分词 ] ...