zoukankan      html  css  js  c++  java
  • boost的初步了解

    本章介绍了 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;

    1. 定义一个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()函数的名字,而该函数本身并没有被调用。

    2. 象 async_wait() 这样的方法被称为是非阻塞式的。 I/O 对象通常还提供了阻塞式的方法,可以让执行流在特定操作完成之前保持阻塞。 例如,可以调用阻塞式的 wait() 方法,取代 boost::asio::deadline_timer 的调用。 由于它会阻塞调用,所以它不需要传入一个函数名,而是在指定时间点或指定时长之后返回。可以留意到在调用 async_wait() 之后,又在 I/O 服务之上调用了一个名为 run() 的方法。这是必须的,因为控制权必须被操作系统接管,才能在五秒之后调用 handler() 函数。
    3. 这里存在一个疑问: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()里的参数是不能丢的!!!

  • 相关阅读:
    React开发实时聊天招聘工具 -第六章 登陆注册(2)
    React开发实时聊天招聘工具 -第六章 登陆注册(1)
    温习 socket http tcp
    高阶组件简介
    计算机组成原理(7)——输入、输出系统
    计算机组成原理(6)——总线
    计算机组成原理(5)——中央处理器
    计算机组成原理(4)——指令系统
    计算机组成原理(3)——存储层次结构
    计算机组成原理(2)——数据的表示与运算
  • 原文地址:https://www.cnblogs.com/boost/p/10430176.html
Copyright © 2011-2022 走看看