在python中从类似cron的调度程序中删除事件

时间:2012-07-20 22:18:44

标签: python cron

所以我按照这里提出的问题的答案建议创建一个类似cron的调度程序:

How do I get a Cron like scheduler in Python?

但有时,如果发生事件,我想删除其他事件......

我如何创建此功能?....基本上,一旦创建,我怎么能删除一个事件?

另外......我在活动中制作活动时遇到了麻烦。基本上,我的目标是每小时使用一个事件来解析文件,这样可以保留我想要做其他事件的时间......所以我想要在这个每小时的事件中创建新事件......但它们似乎在被创造后就消失了......

谢谢!

1 个答案:

答案 0 :(得分:0)

这一切都取决于你实际上如何实现你的cronjob运行。但如果您希望事件能够修改其他事件,我可以考虑3个选项:

1。将对cron的引用传递给函数

您的CronTab课程必须为事件提供一种方法,以便能够检查"进去。最好的选择是使用" cron"扩展关键字参数。参数并将其传递给函数,如下所示:

class CronTab(object):
    def __init__(self, *events):
        self.events = events
        for event in self.events:
             event.kwargs['cron'] = self
# rest of the calss remains unmodified

这样做,从Crontab运行中调用的所有函数都将被传递给" cron"引用正在运行它的cron的参数

2。添加"预运行"编组(S)

扩展CronTab类不仅要执行特定计划中的事件,还要在这些事件之前调用其他函数。获取事件实例的函数,必须返回True才能运行它。如果其中任何一个返回其他内容,则相关事件将无法运行。

基本(和未经测试)的实现将是:

class CronTab(object):
    def __init__(self, *events, marshalls=None):
        self.events = events
        if marshalls is not None:
            # marshalls must be a list of callables that return True or False
            self.marshalls = marshalls
        else
            self.marshalls = []

    def run(self):
        t=datetime(*datetime.now().timetuple()[:5])
        while 1:
            for e in self.events:
                if all([x(e) for x in self.marshalls]):
                    e.check(t)

            t += timedelta(minutes=1)
            while datetime.now() < t:
                time.sleep((t - datetime.now()).seconds)

显然有更好的方法来实现这一点。例如,通过执行reduce操作,从而避免在其中一个事件已经返回False之后为所有事件运行所有marshalls。但这应该指向正确的方向。

如果CronTab类实际上做的工作多于调用事件的实例check,或者至少将运行检查与实际运行事件分开,那么实现它也可能更容易 - 但我不想修改原始代码那么多。

3。提供事件挂钩

这与以前的解决方案基本相同,但更多的细节&#34;。您可以拥有多个列表,而不只是拥有一个预先运行的元帅列表。确切的数量和细节可以根据您的需求量身定制,但根据您的要求,这将是我的选择。

除了一般&#34;预先运行的marshalls&#34;你可以为每个事件名称都有一个这样的列表。在举办活动之前,您不仅可以运行所有普通的&#34;预运行的组织&#34;还有特定于事件的那些。

实现这一目标的一种方法是制作&#34;预先运行的元帅&#34;列入列表字典并提供一些管理此类列表的方法。像这样:

class CronTab(object):
    def __init__(self, *events, marshalls=None):
        self.events = events
        self.marshalls = dict()
        if marshalls is not None:
            # marshalls must be a list of callables that return True or False
            self.marshalls['__general'] = marshalls
        else
            self.marshalls['__general'] = []

    # event is a string with the name of the event you want to marshall
    # or None if you want to marshall all of them 
    def register(self, callback, event=None):
        if event = None:
            self.marshalls['__general'].append(callback)
        else:
            self.marshalls.setdefault(event, list()).append(callback)

    def run(self):
        t=datetime(*datetime.now().timetuple()[:5])
        while 1:
            for e in self.events:
                if all([x(e) for x in self.marshalls['__general']]) and
                   all([x(e) for x in self.marshalls[e.name]]):
                    # the above assumes e.name exists (not in reference implementation)
                    e.check(t)

        t += timedelta(minutes=1)
        while datetime.now() < t:
            time.sleep((t - datetime.now()).seconds)

以同样的方式,您可以根据需要添加运行后挂钩和其他奇怪的东西。


关于有关活动消失的问题,请向我们展示一些代码,以便进一步解决问题。