boost::asio基本使用
一、Asio网络库
截止到C++17,C++标准库都没有加入网络通信库。实际项目网络编程是非常常见的功能,直接使用操作系统API是低效率且不稳定的,比较好的方法是借助第三方成熟可靠的网络库。据我所知C++中目前比较有名的网络库有ACE、libevent和boost.Asio,这三个库都是跨平台的,各有特色,对于一般的应用来说,使用这些库都是没什么问题。由于C++标准库和boost库的亲缘关系,我经常使用boost中其它的库,所以选择了Asio库。在网络上搜索了大家对Asio的评价,发现Asio库作为跨平台的网络库还是相当优秀的。
Linux系统高效网络I/O是epoll,windows系统高效的网络I/O是iocp,epoll是一种同步I/O复用技术,iocp是异步I/O。同步I/O指内核通知应用程序数据有了,应用程序可以获取了;异步I/O是内核负责读数据,读好后通知应用程序可以用了。所以一般来说异步I/O相比同步I/O的效率更高一些,但是Linux中异步I/O(aio)并不比epoll高效。为了使的Asio库跨平台,最终Asio选择Linux系统在epoll的基础上用iocp的思想的封装一层,这在一定上损失了Linux平台I/O的效率。总的来说Asio网络通信库是主动器模式(proactor),而libevent使用的是reactor模式。
二、反应器模式/主动器模式
反应器是比较容易理解的I/O模式,也是使用比较多的模式。主要内容来自于《IO设计模式:Reactor和Proactor对比》。

Reactor包含如下角色:
- Handle 句柄;用来标识socket连接或是打开文件;
 - Synchronous Event Demultiplexer:同步事件多路分解器:由操作系统内核实现的一个函数;用于阻塞等待发生在句柄集合上的一个或多个事件;(如select/epoll;)
 - Event Handler:事件处理接口
 - Concrete Event HandlerA:实现应用程序所提供的特定事件处理逻辑;
 - Reactor:反应器,定义一个接口,实现以下功能: 
1)供应用程序注册和删除关注的事件句柄;
2)运行事件循环;
3)有就绪事件到来时,分发事件到之前注册的回调函数上处理; 

Proactor主动器模式包含如下角色
- Handle 句柄;用来标识socket连接或是打开文件;
 - Asynchronous Operation Processor:异步操作处理器;负责执行异步操作,一般由操作系统内核实现;
 - Asynchronous Operation:异步操作
 - Completion Event Queue:完成事件队列;异步操作完成的结果放到队列中等待后续使用
 - Proactor:主动器;为应用程序进程提供事件循环;从完成事件队列中取出异步操作的结果,分发调用相应的后续处理逻辑;
 - Completion Handler:完成事件接口;一般是由回调函数组成的接口;
 - Concrete Completion Handler:完成事件处理逻辑;实现接口定义特定的应用处理逻辑;
 
主动和被动以主动写为例,Reactor将handle放到select(),等待可写就绪,然后调用write()写入数据;写完处理后续逻辑;Proactor调用aoi_write后立刻返回,由内核负责写操作,写完后调用相应的回调函数处理后续逻辑;可以看出,Reactor被动的等待指示事件的到来并做出反应;它有一个等待的过程,做什么都要先放入到监听事件集合中等待handler可用时再进行操作;Proactor直接调用异步读写操作,调用完后立刻返回。
三、基本使用
1.基本的同步客户端架构
using boost::asio;
io_service service;
ip::tcp::endpoint ep( ip::address::from_string("127.0.0.1"), );
ip::tcp::socket sock(service);
sock.connect(ep);
2.基本的同步服务器架构
typedef boost::shared_ptr<ip::tcp::socket> socket_ptr;
io_service service;
ip::tcp::endpoint ep( ip::tcp::v4(), )); // listen on 2001
ip::tcp::acceptor acc(service, ep);
while ( true) {
socket_ptr sock(new ip::tcp::socket(service));
acc.accept(*sock);
boost::thread( boost::bind(client_session, sock));
}
void client_session(socket_ptr sock) {
while ( true) {
char data[];
size_t len = sock->read_some(buffer(data));
if ( len > )
write(*sock, buffer("ok", ));
}
3.基本的异步客户端架构
using boost::asio;
io_service service;
ip::tcp::endpoint ep( ip::address::from_string("127.0.0.1"), );
ip::tcp::socket sock(service);
sock.async_connect(ep, connect_handler);
service.run();
void connect_handler(const boost::system::error_code & ec) {
// 如果ec返回成功我们就可以知道连接成功了
}
4.基本的异步服务器架构
using boost::asio;
typedef boost::shared_ptr<ip::tcp::socket> socket_ptr;
io_service service;
ip::tcp::endpoint ep( ip::tcp::v4(), )); // 监听端口2001
ip::tcp::acceptor acc(service, ep);
socket_ptr sock(new ip::tcp::socket(service));
start_accept(sock);
service.run();
void start_accept(socket_ptr sock) {
acc.async_accept(*sock, boost::bind( handle_accept, sock, _1) );
}
void handle_accept(socket_ptr sock, const boost::system::error_code &
err) {
if ( err) return;
// 从这里开始, 你可以从socket读取或者写入
socket_ptr sock(new ip::tcp::socket(service));
start_accept(sock);
}
boost::asio基本使用的更多相关文章
- c++ boost asio库初学习
		
前些日子研究了一个c++的一个socket库,留下范例代码给以后自己参考. 同步server: // asio_server.cpp : コンソール アプリケーションのエントリ ポイントを定義します. ...
 - 如何在多线程leader-follower模式下正确的使用boost::asio。
		
#include <assert.h> #include <signal.h> #include <unistd.h> #include <iostream& ...
 - BOOST.Asio——Tutorial
		
=================================版权声明================================= 版权声明:原创文章 谢绝转载 啥说的,鄙视那些无视版权随 ...
 - BOOST.Asio——Overview
		
=================================版权声明================================= 版权声明:原创文章 谢绝转载 啥说的,鄙视那些无视版权随 ...
 - boost asio sync
		
Service: #include<boost/asio.hpp> #include<boost/thread.hpp> #include<iostream> #i ...
 - 网络库crash以及boost asio strand dispath分析
		
最近在做服务器的稳定性的相关测试,服务器的网络底层使用的是boost asio,然后自己做的二次封装以更好的满足需求. 服务器昨天晚上发现crash了一次,之前测试了将近半个多月,有一次是莫名的退出了 ...
 - boost asio tcp server 拆分
		
从官方给出的示例中对于 boost::asio::ip::tcp::acceptor 类的使用,是直接使用构造函数进行构造对象,这一种方法用来学习是一个不错的方式. 但是要用它来做项目却是不能够满足我 ...
 - boost::asio 使用实例
		
#include <iostream> #include <boost/asio.hpp> using namespace std; using namespace boost ...
 - boost.asio源码剖析(四) ---- asio中的泛型概念(concepts)
		
* Protocol(通信协议) Protocol,是asio在网络编程方面最重要的一个concept.在第一章中的levelX类图中可以看到,所有提供网络相关功能的服务和I/O对象都需要Protoc ...
 - boost.asio源码剖析(五) ---- 泛型与面向对象的完美结合
		
有人说C++是带类的C:有人说C++是面向对象编程语言:有人说C++是面向过程与面向对象结合的语言.类似的评论网上有很多,虽然正确,却片面,是断章取义之言. C++是实践的产物,C++并没有为了成为某 ...
 
随机推荐
- jquery 问题
			
detach():这个方法不会把匹配的元素从jQuery对象中删除,因而可以在将来再使用这些匹配的元素.与remove()不同的是,所有绑定的事件.附加的数据等都会保留下来. jquery ajax不 ...
 - Spring注解使用和与配置文件的关系
			
Spring注解使用和与配置文件的关系 1 注解概述与容器管理机制 Spring 2.5 中除了提供 @Component 注释外,还定义了几个拥有特殊语义的注释,它们分别是:@Repositor ...
 - Android Sqlite 增删改查
			
模拟 查询所有数据,增加一条数据,修改某一条数据,删除某一条数据: SQLiteOpenHelper 帮助类的介绍: import android.content.Context; import an ...
 - C#使用cplex求解简单线性规划问题(Cplex系列-教程二)
			
若还未在项目中添加cplex的引用,可以参阅上一篇文章.本文主要介绍利用C#求解线性规划的步骤,对线性规划模型进行数据填充的两种方法,以及一些cplex函数的功能和用法.包括以下几个步骤: 描述 先花 ...
 - DotNetty 使用ByteToMessageDecoder 国家部标808协议封装
			
DotNetty 开源地址 https://github.com/Azure/DotNetty 个人博客地址 http://www.dncblogs.cn/Blog/ShowBlog/70 1.国 ...
 - (C#)调用Webservice,提示远程服务器返回错误(500)内部服务器错误
			
因为工作需要调用WebService接口,查了下资料,发现添加服务引用可以直接调用websevice 参考地址:https://www.cnblogs.com/peterpc/p/4628441.ht ...
 - JavaScript编程题(含腾讯2016校招题)
			
作者:ManfredHu 链接:http://www.manfredhu.com/2016/04/02/15-veryGoodForUsing/ 声明:版权所有,转载请保留本段信息,否则请不要转载 几 ...
 - 201621123018《Java程序设计》第4周学习报告
			
1. 本周学习总结 1.1 写出你认为本周学习中比较重要的知识点关键词 继承.多态.子类.父类.抽象类.abstract.super.object 1.2 尝试使用思维导图将这些关键词组织起来.注:思 ...
 - bzoj4519: [Cqoi2016]不同的最小割(最小割树)
			
传送门 好神仙……最小割树是个什么东西…… 其实我觉得干脆直接$O(n^2)$跑几个dinic算了…… 来说一下这个叫最小割树的神奇东西 我们先建一个$n$个点,没有边的无向图 在原图中任选两点$s, ...
 - 浅谈对MVC的认识
			
MVC是model(模型),view(视图),Controller(控制)的缩写. 模型层负责提供数据,和数据库相关的操作都交给模型层处理: 视图层提供交互的界面,其输出数据: 控制层负责接收各种请求 ...