本章介绍了 Boost C++ 库 Asio,它是异步输入输出的核心。 名字本身就说明了一切:Asio 意即异步输入/输出。 该库可以让 C++ 异步地处理数据,且平台独立。 异步数据处理就是指,任务触发后不需要等待它们完成。 相反,Boost.Asio 会在任务完成时触发一个应用。 异步任务的主要优点在于,在等待任务完成时不需要阻塞应用程序,可以去执行其它任务。
异步任务的典型例子是网络应用。 如果数据被发送出去了,比如发送至 Internet,通常需要知道数据是否发送成功。 如果没有一个象 Boost.Asio 这样的库,就必须对函数的返回值进行求值。 但是,这样就要求待至所有数据发送完毕,并得到一个确认或是错误代码。 而使用 Boost.Asio,这个过程被分为两个单独的步骤:第一步是作为一个异步任务开始数据传输。 一旦传输完成,不论成功或是错误,应用程序都会在第二步中得到关于相应的结果通知。 主要的区别在于,应用程序无需阻塞至传输完成,而可以在这段时间里执行其它操作。
使用 Boost.Asio 进行异步数据处理的应用程序基于两个概念:I/O 服务和 I/O 对象。 I/O 服务抽象了操作系统的接口,允许第一时间进行异步数据处理,而 I/O 对象则用于初始化特定的操作。 鉴于 Boost.Asio 只提供了一个名为 boost::asio::io_service 的类作为 I/O 服务,它针对所支持的每一个操作系统都分别实现了优化的类,另外库中还包含了针对不同 I/O 对象的几个类。 其中,类 boost::asio::ip::tcp::socket 用于通过网络发送和接收数据,而类 boost::asio::deadline_timer 则提供了一个计时器,用于测量某个固定时间点到来或是一段指定的时长过去了。 以下第一个例子中就使用了计时器,
在CMakeLists.txt里边要这么定义boost库,否则编译器找不见boost库
cmake_minimum_required(VERSION 3.8) project(boost_timer) set(CMAKE_CXX_STANDARD 11) set(SOURCE_FILES main.cpp) find_package(Boost REQUIRED COMPONENTS system) if(Boost_FOUND) include_directories(${Boost_INCLUDE_DIRS}) add_executable(boost_timer ${SOURCE_FILES}) target_link_libraries(boost_timer Boost::system ws2_32) endif()
在main.cpp这么写:
#include <iostream> #include <boost/asio.hpp> void handly(const boost::system::error_code &ec) { std::cout << "5s" << std::endl; } int main() { boost::asio::io_service _io_service; boost::asio::deadline_timer timer(_io_service,boost::posix_time::seconds(5)); timer.async_wait(handly); _io_service.run(); return 0; }
首先boost提供的I/O服务只有一个就是boost::asio::io_serivce _io_service . 在这里的I/O对象是boost::asio::deadline_timer timer;
- 定义一个I/O服务,_io_service , 再定义一个I/O对象timer , 通常I/O对象的构造函数的第一个参数就是I/O服务 _io_service , 由于timer是一个定时器,所以就是有第二个参数boost::posix_time::second(int time);用于表示time时间段 , 该闹钟在 timer 被定义之后立即开始计时。async_wait() 的好处是,该函数调用会立即返回,而不是等待五秒钟。 一旦闹钟时间到,作为参数所提供的函数就会被相应调用。 因此,应用程序可以在调用了 async_wait() 之后执行其它操作,而不是阻塞在这里。请留意,我们只是传入了handly()函数的名字,而该函数本身并没有被调用。
- 象 async_wait() 这样的方法被称为是非阻塞式的。 I/O 对象通常还提供了阻塞式的方法,可以让执行流在特定操作完成之前保持阻塞。 例如,可以调用阻塞式的 wait() 方法,取代 boost::asio::deadline_timer 的调用。 由于它会阻塞调用,所以它不需要传入一个函数名,而是在指定时间点或指定时长之后返回。可以留意到在调用 async_wait() 之后,又在 I/O 服务之上调用了一个名为 run() 的方法。这是必须的,因为控制权必须被操作系统接管,才能在五秒之后调用 handler() 函数。
- 这里存在一个疑问:async_wait()是非阻塞的但是io_service::run()却是阻塞的呢?这是由于应用程序必须防止被中止执行,所以这样做实际上不会有任何问题。 如果 run() 不是阻塞的,main() 就会结束从而中止该应用程序。 如果应用程序不应被阻塞,那么就应该在一个新的线程内部调用 run(),它自然就会仅仅阻塞那个线程。也就是说如果没有run()函数这个阻塞的话,程序运行到 timer.async_wait(handly);然后就直接return 0;结束了
#include <iostream> #include <boost/asio.hpp> void handly1(const boost::system::error_code &ec) { std::cout << "5s" << std::endl; } void handly2(const boost::system::error_code & ec) { std::cout << "10s" << std::endl; } int main() {
boost::asio::io_service _io_service; boost::asio::deadline_timer timer1(_io_service,boost::posix_time::seconds(5)); timer1.async_wait(handly1); boost::asio::deadline_timer timer2(_io_service,boost::posix_time::seconds(10)); timer2.async_wait(handly2); _io_service.run();
return 0; }
结果就是:程序运行5秒后调用handly1()函数;
程序运行10秒后调用handly2()函数;
注意handly1()和handly2()里的参数是不能丢的!!!