Asio的架构:Boost.Asio 设计索引
概念性了解API:boost::asio中的同步与异步
Asio的Buffer: buffer几种用法,这些Buffer都只是引用外部的内存数据,如果需要拷贝和分配,记得使用boost::pool,这里还有一篇处理拷贝Buffer的文章
例子解析: Boost.asio的简单使用(timer,thread,io_service类)
如果照着例子弄出的第一个服务器无法收到客户端消息,试试这个asio::async_read与socket的async_read_some的区别
这里是另外一个区别:boost.asio库学习笔记—— receive和read的区别:
从服务器连接过来的客户端的地址:
std::string endpoint = socket.remote_endpoint( ).address( ).to_string();
以下是对这篇文章的翻译:
asio chat_client.cpp中的一些问题
1. 有多少个线程在运行2个,还是3个?
>一般来说,依赖于运行的平台,从程序的角度来说是2个,包括:
*主线程,用于处理用户的输入输出
*io_service.run()线程,用于处理chat_client对象中的所有行为(action)
还有,async_write会创建一个线程或者其他的一些东西么?
>不会.
2. 有关1的问题,为什么write函数使用post直接调用?什么不调用async_write?既然调用了post,你只是将其放到一个队列里在同一线程处理,为什么之后还要从其他线程调用async_write?
chat_client的成员对象不是线程安全的(故意?),因此要同步处理这些成员。如果直接从主线程调用async_write不是线程安全的,因为此时可能有后台线程正在访问socket。
在这个例子中,所有的类成员都调用io_services.post()以保证在一个线程里访问,达到线程安全。io_services保证任何使用io_services.post()(或io_servies.dispatch())传入的句柄只会在io_serive.run()线程被调用。而且这个例子中只有一个线程调用io_service.run(),所以chat_client的成员变量也只会在一个线程中被访问。
4. 如果我想发送一个连接事件到主线程,怎样做?用io_service::post?能从主线程获取io_services?
在这个例子里是很困难的,因为主线程正在阻塞等待用户信息。不过如果你想将事件在线程间传递,确实可以为每个线程配备一个io_services。
5. 为什么在main函数的最后调用了t.join(),能用io_service.run()代替么?
不行,请参考问题2的解答,那样的话,线程安全将无法保证
6. 按照问题1的解答,如果有3个线程在运行(也就是,async_write被放到另外一个线程),那么哪个位置创建这个线程比较好?
因为主线程需要阻塞等待用户信息,因此io_service::run是唯一需要的。如果你的程序不需要这样做,那么就不需要其他线程,也就只需要简单的调用io_service::run()就可以了,这也是大多数例子这样做的原因
有关线程安全的问题
1. 对于asio对象,能从2个不同线程调用一个共享对象的不同成员么?
不能
那么其意义就是从2个不同线程访问共享对象不是线程安全的?
是的
只有被标记为 “共享对象:安全”的对象才能从不同线程同时访问,io_service就是这样的对象
2. 同样是线程安全的问题,对于basic_deadline_timer::cancel()我需要用io_service.post(boost::bind(&deadline_timer::cancel, &myTimer))方法封装调用么?
是的,直接调用cancel()也不是线程安全的
最好的解决方法就是使用io_service::post()将所有的操作都放在一个线程
3. asio有很多成员函数,我怎么知道哪些能安全的调用?
一般情况下,你应该认为没有任何一个函数是安全的,以下是通用的io线程安全判断用例:
write+write:不安全
read+write:不安全
read+read:安全
asio对象已经符合这种需求
这里有一篇介绍io_service众多区别及包处理,拆包等的技术