1. 基础熟悉
muduo中执行定时任务有三个方法:EventLooprunAt
、runAfter
和runEvery
。
runAt
:某个时间执行任务
runAfter
:某段时间后执行任务
runEvery
:定时循环执行任务
这两个函数的原型:
TimerId runAt(const Timestamp& time, TimerCallback&& cb);
TimerId runAfter(double delay, TimerCallback&& cb);
TimerId runEvery(double interval, TimerCallback&& cb);
2. 问题
从上面的函数原型可以看出:这三个函数都返回了一个TimerId
对象,TimerId
很简单:
class Timer;
///
/// An opaque identifier, for canceling Timer.
///
class TimerId : public muduo::copyable
{
public:
TimerId() : timer_(NULL), sequence_(0) {}
TimerId(Timer* timer, int64_t seq) : timer_(timer), sequence_(seq) {}
// default copy-ctor, dtor and assignment are okay
friend class TimerQueue;
private:
Timer* timer_;
int64_t sequence_;
};
作者陈硕 说An opaque identifier, for canceling Timer.
,意思说这是一个用于取消任务的不透明的标识
,确实不透明,两个成员变量外界都不能访问,除非友元类TimerQueue
。
执行上面三个方法发起一个定时任务,得到TimerId标识
,该标识是一个对象。我个人感觉很不方便,我们项目中很多数据存储在redis中,TimerId标识
不能直接存储字redis中,需要自己在内存中保存这个对象的映射。
当然自己比较菜鸟,不能理解muduo作者的真正意图。我希望返回的是一个整型,或者字符串类型的标志,这样外界保存和使用都非常方便。
3. 解决方案
基于上面我说的,TimerId标识
很难存储(至少在我的项目中),我不得不考虑在取消一个任务
或不取消任务,在任务执行时判断是否生效
两个方案的选择。
不取消任务,在任务执行时判断是否生效
是指我在需要取消任务时,不去取消任务,可以节省TimerId的处理,发起一个任务就不用管了,直到任务执行的时候,来判断是否还需要再真正执行该任务。
这是一个很丑的方法,但是比起对TimerId标识
的处理,简单了很多。
希望指正,有更好的方式!