BOOST 定时器 stop探究
Asio是一个建立在Boost所提供的相关组件之上的异步的网络库,可以运行在Win/Linux/Unix等各种平台之上。
不过随着C++11的发布,其对Boost的依赖也越来越少,作者又做了一个不依赖于Boost的版本。
1. 同步Timer: 调用wait后立即阻塞
#include <iostream>
#include <boost/asio.hpp> int main()
{
boost::asio::io_service io;
boost::asio::deadline_timer timer(io, boost::posix_time::seconds(5)); timer.wait(); std::cout << "Hello, world!" << std::endl; return 0;
}
实测效果,程序开始运行后3秒,打印了:Hello, world!
2. 异步Timer: wait后不立即阻塞,调用io的run方法时才会阻塞
#include <iostream>
#include <boost/asio.hpp>
#include <boost/thread.hpp> void print(const boost::system::error_code& err)
{
std::cout << "Hello, world! " << std::endl;
} int main()
{
boost::asio::io_service io;
boost::asio::deadline_timer timer(io, boost::posix_time::seconds(5)); std::cout << "----gointo 0---- " << std::endl; timer.async_wait(&print);
//timer.cancel(); 取消定时器, 这样后续io.run方法将不会阻塞,但是定时器回调函数会立马被执行 std::cout << "----gointo 1---- " << std::endl; io.run();
std::cout << "----gointo 2-!--- " << std::endl; return 0;
}
运行:
2.1
如果让上图内的timer.cancel()这句代码工作,那么效果将是:
取消定时器后, 定时器回调函数会立马被执行,同时后续io.run方法将不会阻塞 !
2.2
如果我们希望在开启定时器后,本线程内的代码流仍能继续向下走,
我们只有改一改代码了:将开启定时器的代码封装在一个新的线程内,这样阻塞的就不是本线程,而是新开的线程了。
如下面的例子3所示。
3 异步Timer: 将开启定时器的代码封装在一个新的线程内,这样阻塞的就不是本线程,而是新开的线程了。
#include <iostream>
#include <boost/asio.hpp>
#include <boost/thread.hpp> void print(const boost::system::error_code& err)
{
std::cout << "Hello, world! " << std::endl;
} void func_timer_test()
{
boost::asio::io_service io;
boost::asio::deadline_timer timer(io, boost::posix_time::seconds(5)); std::cout << "----gointo 0---- " << std::endl;
timer.async_wait(&print);
std::cout << "----gointo 1---- " << std::endl; io.run();
std::cout << "----gointo 2---- " << std::endl;
} int main()
{
boost::thread thread_timer_test(func_timer_test);
thread_timer_test.detach(); while(1){
std::cout << "----main loop()--- " << std::endl;
boost::this_thread::sleep(boost::posix_time::seconds(1));
} return 0;
}
运行:
4 异步Timer: 取消定时器,因为取消定时器后,事先设置的定时器回调函数会被立马执行,所以必须在定时器回调函数内通过当前的错误码来进行判断。
#include <iostream>
#include <boost/asio.hpp>
#include <boost/thread.hpp>
#include <boost/exception/all.hpp>
#include <boost/system/error_code.hpp>
#include <boost/asio.hpp>
#include <string> void print(const boost::system::error_code& err)
{
if(err){
std::cout << err.category().name() << std::endl;
std::cout << err.value() << std::endl;
} std::cout << "Hello, world! " << std::endl;
} int main()
{
boost::asio::io_service io;
boost::asio::deadline_timer timer(io, boost::posix_time::seconds(5)); std::cout << "----gointo 0---- " << std::endl; timer.async_wait(&print);
timer.cancel(); std::cout << "----gointo 1-*--- " << std::endl; io.run(); std::cout << "----gointo 2-$--- " << std::endl; return 0;
}
4.1
如果屏蔽上图代码内的timer.cancel()这句,即不取消定时器时,
运行结果如下图:
存在5秒的定时。
4.2
取消定时器,即上图代码的运行结果:
代码立即执行完毕,且定时器回调内判断出,发生了异常。
那么,我们平常在使用boost库定时器的时候,如果我们希望真正取消一个事先已经设置好的定时器回调函数的执行,
我们可以换种思路:在定时器回调函数内做if-else判断,如果被取消了干啥(一般是空处理),定时结束正常进入时干啥,区分开处理就好了。
后记:
对于取消定时器任务,如果我在主线程内调用timer.cancel来取消定时器线程内的定时器任务,实测运行会段错误。
也即是说,本博客最终还是没能搞定单次定时器任务的取消。
百度了不少博客,都没能搞定定时器任务的取消。
需要重要说明的是:我所谓的取消定时器任务的严格定义是,我开了一个定时器任务,但是这个任务由于时间未到所以还没有被执行到,我中途希望取消该定时器任务。
https://blog.csdn.net/zhongjiezheng/article/details/44343959
这篇博客的访问量竟然还挺高的,但是他实现的是一个周期性定时器,他所谓的停止指的仅仅是取消这个定时器任务的周期性执行而已。和我所需要的取消定时器任务的需求完全是两码事。
我们来验证上面博客的代码是无法完成取消定时器任务的,复制其代码,简单修改了下,贴代码:
#include <boost/asio.hpp>
#include <boost/thread.hpp>
#include <iostream> class MyTimer
{
public:
template<typename F>
MyTimer(boost::asio::io_service& ios, F func, int interval, bool start = true) :
m_func(func),
m_interval(interval),
m_timer(ios, boost::posix_time::millisec(interval)),
m_flag(start)
{
m_timer.async_wait(boost::bind(&MyTimer::call_func, this, boost::asio::placeholders::error));
}
void call_func(const boost::system::error_code&)
{
if (!m_flag)
{
std::cout << "cancel!" << std::endl;
return;
}
m_func();
//m_timer.expires_at(m_timer.expires_at() + boost::posix_time::millisec(m_interval));
//m_timer.async_wait(boost::bind(&MyTimer::call_func, this, boost::asio::placeholders::error));
}
void stop()
{
m_flag = false;
}
void start(int new_interval)
{
m_flag = true;
m_interval = new_interval;
m_timer.expires_from_now(boost::posix_time::millisec(m_interval));
m_timer.async_wait(boost::bind(&MyTimer::call_func, this, boost::asio::placeholders::error));
}
private:
boost::function<void()> m_func;
int m_interval;
boost::asio::deadline_timer m_timer;
bool m_flag;
}; void print1()
{
std::cout << "11111" << std::endl;
} void print2()
{
std::cout << "22222" << std::endl;
} boost::asio::io_service ios;
MyTimer at1(ios, print1, 15000); // 希望该定时器任务15秒后被执行,中途我们来取消之
//MyTimer at2(ios, print2, 8000); void threadfun()
{
boost::this_thread::sleep(boost::posix_time::milliseconds(2000));
at1.stop(); //at2.start(1000); //at1.stop();
//at2.stop();
} int main()
{
// boost::this_thread::sleep(boost::posix_time::milliseconds(5000));
boost::thread_group thr_grp;
thr_grp.create_thread(boost::bind(threadfun));
ios.run();
getchar();
return 0;
} // 实验失败 定时器任务没有被取消。
运行:
实验失败 定时器任务没有被取消。
好吧,折腾得够久了,无比的烦躁。 我还是自己基于linux去封装一个CPP的定时器吧!
.
.
BOOST 定时器 stop探究的更多相关文章
- boost 定时器.
#include <iostream> #include <boost/asio.hpp> int main() { boost::asio::io_service io; b ...
- 简单的几个Boost定时器
boost的asio库里有几个定时的器,老的有 deadline_timer , 还有三个可配合 C++11 的 chrono 使用的 high_resolution_timer . steady_ ...
- boost timer 定时器 Operation cancel !
前面段时间处理一个定时器时,老是提示 操作取消. 硬是没搞明白为什么! 其实我遇到的这个情况很简单就是(boost::asio::deadline_timer timer)这个变量的生命同期结束了,对 ...
- 使用boost的deadline_timer实现一个异步定时器
概述 最近在工作上需要用到定时器,然后看到boost里面的deadline_timer可以实现一个定时器,所以就直接将其封装成了ATimer类,方便使用,ATimer有以下优点: 可以支持纳秒.毫秒. ...
- boost库学习随记六:使用同步定时器、异步定时器、bind、成员函数回调处理、多线程的同步处理示例等
一.使用同步定时器 这个示例程序通过展示如何在一个定时器执行一个阻塞等待. //makefile #-------------------------------------------------- ...
- [Boost基础]并发编程——asio网络库——定时器deadline_timer
asio库基于操作系统提供的异步机制,采用前摄器设计模式(Proactor)实现了可移植的异步(或者同步)IO操作,而且并不要求使用多线程和锁定,有些的避免了多线程编程带来的诸多有害副作用(如条件竞争 ...
- boost asio 学习(六) 定时器
http://www.gamedev.net/blog/950/entry-2249317-a-guide-to-getting- started-with-boostasio?pg=7 6 定时器 ...
- 使用boost线程定时器作为后台线程来切换主循环程序状态方法2
上一篇的方法主要使用的是:通过线程延时实 现的定时,并且只能定时一次,如果需要对此定时处理,就需要使用下面的定时器: #include "stdafx.h" #include &l ...
- Tcp/ip实验准备:一个简单的定时器——boost实现
tcp/ip实验须要在指定的时间查看结果,为了实验方便,做了一个定时器.用法是: 在命令行输入:timer 输入数字之后,计时对应秒数 输入m数字之后.计时对应分钟数(支持小数分钟数) 输入q退出. ...
- boost::asio学习(定时器)
#include <boost/asio.hpp> #include <iostream> void handle1(const boost::system::error_co ...
随机推荐
- oeasy教您玩转vim - 44 - # 综合练习
综合练习 回忆上节课内容 上次我们学到了各种的替换模式 r,替换当前光标的字符 R,进入替换模式 ~,改变大小写 这次来个综合练习吧! 我们这次要完成这样一个任务 我们有的时候需要写日志 Syst ...
- oeasy教您玩转vim - 68 - # 标签页tab
tab选项卡 回忆上次 上次有三种批量替换,分别是 :windo :bufdo :argdo 执行的{cmd}可以用|按顺序增加 分别对应的 windows 窗口 buffers 缓存文件 arg ...
- iOS开发基础136-防暴力点击
要在Objective-C中创建一个高度可复用的工具类,以防止按钮的暴力点击,并且使用切面编程(AOP)的方式,我们可以考虑使用Aspects这个库来实现方法的拦截.以下是具体的实现步骤: 第一步:引 ...
- 单细胞测序最好的教程(十四)测序原始数据公开至NCBI数据库
作者按 国内对于单细胞测序相关的中文教程确实不够全面,当然NCBI官网给的上传教程也比较详细了,所以变成了会者不难.本教程你现在可能用不上,但是你如果做单细胞测序,那么未来你一定会用上,建议收藏. 在 ...
- 搭建lnmp环境-php(第二步)
系统环境:centos7 php7.4 编译安装太繁琐,这里用yum安装即可 ===========yum形式安装======== # 安装EPEL源(nginx那里已安装了,跳过) yum inst ...
- java引入es使用
引入依赖 <dependency> <groupId>org.elasticsearch.client</groupId> <artifactId>el ...
- uniapp打包所需的ios证书和证书profile文件获取的图文教程
使用uniapp进行云打包,可以打包android和ios两种app,但是uniapp官方并不能凭空产生这两种平台所需的打包证书. 那么这两种打包证书又是如何获取呢? android相对简单,使用jd ...
- 【Docker】03 基础操作
[Docker 本体操作相关] 检查Docker版本: docker -v 检查Docker当前状态: systemctl status docker 停止Docker与开启Docker system ...
- 人形机器人(humanoid)的摔倒实验/撞击实验
motivation: 人形机器人的实体比较昂贵,但是实验过程中机器人的摔倒和撞击时十分常见的事情,这就会导致机器人的元器件被损坏,造成较大的经济损失,为此我们在设计机器人的机械结构和电子元器件布局时 ...
- 国产计算框架mindspore在gpu环境下1.3.0版本的分布式计算组件安装 ——(openmpi 和 nccl 的安装,配置,示例代码的运行)
前文已经给出1.3.0gpu版本的编译及安装,本文在此基础上进行分布式组件的安装,前文信息参看: 国产计算框架mindspore在gpu环境下编译分支r1.3,使用suod权限成功编译并安装,成功运行 ...