如何防止 cron 作业执行(如果它已经在运行)

Posted

技术标签:

【中文标题】如何防止 cron 作业执行(如果它已经在运行)【英文标题】:How to prevent the cron job execution, if it is already running 【发布时间】:2012-05-20 02:08:09 【问题描述】:

我有一个 php 脚本,我在 CentOS 上每 10 分钟通过 cron 执行这个脚本。

问题在于,如果 cron 作业需要超过 10 分钟,那么同一 cron 作业的另一个实例将启动。

我尝试了一个技巧,那就是:

    使用 php 代码创建了一个锁定文件(与 pid 文件相同) cron 作业开始了。 在作业完成时删除了带有 php 代码的锁定文件。 当任何新的 cron 作业开始执行脚本时,我检查了是否锁定 文件存在,如果存在,则中止脚本。

但是可能存在一个问题,即当锁定文件由于某种原因没有被脚本删除或删除时。 cron 将永远不会重新启动。

如果 cron 作业已经在运行,我有什么办法可以再次停止它的执行,使用 Linux 命令或类似的命令?

【问题讨论】:

(编辑)我的建议只是作为一种解决方案发布,以防flock() 不起作用,我建议您先尝试Jack 的解决方案。 我已经测试过使用flock() 但由于某种原因并没有停止 cron 脚本的多个实例。所以经过一番搜索,我找到了这个类并稍微修改了它以供我自己使用:abhinavsingh.com/blog/2009/12/… 使它成为一个服务:forever() do_stuff(); sleep_10_minutes(); (伪代码) @Leonid Shagabutdinov 的回答应该被认为是解决这个问题的正确方法。 【参考方案1】:

咨询锁定正是为此目的而设计的。

您可以使用flock() 完成建议锁定。只需将该函数应用于先前打开的锁定文件,以确定另一个脚本是否对其进行了锁定。

$f = fopen('lock', 'w') or die ('Cannot create lock file');
if (flock($f, LOCK_EX | LOCK_NB)) 
    // yay

在这种情况下,我添加了LOCK_NB 以防止下一个脚本等到第一个脚本完成。由于您使用的是 cron,因此总会有下一个脚本。

如果当前脚本提前终止,任何文件锁都会被操作系统释放。

【讨论】:

感谢杰克的好回答。这是最好的解决方案。但我只是担心如果 cron 因任何原因停止(可能是电源故障或系统重启)并且无法释放锁定。比 cron 不会再次启动,因为它无法再次获取锁。我想我应该先试试这个:) 不可能。无论进程何时停止,操作系统都会释放锁定,为此 h00ligan 的链接文章需要一些丑陋的本地化破解。 这应该是公认的答案。它完全按照我的预期工作。谢谢分配杰克。【参考方案2】:

如果可以配置,或许最好不要写代码:

https://serverfault.com/questions/82857/prevent-duplicate-cron-jobs-running

【讨论】:

【参考方案3】:

flock() 对我来说效果很好 - 我有一个每 5 分钟安排一次数据库请求的 cron 作业,因此不能同时运行多个请求至关重要。这就是我所做的:

$filehandle = fopen("lock.txt", "c+");

if (flock($filehandle, LOCK_EX | LOCK_NB)) 
    // code here to start the cron job
   flock($filehandle, LOCK_UN);  // don't forget to release the lock
 else 
    // throw an exception here to stop the next cron job


fclose($filehandle);

如果您不想终止下一个计划的 cron 作业,而只是暂停它直到正在运行的作业完成,那么只需省略 LOCK_NB

if (flock($filehandle, LOCK_EX)) 

【讨论】:

【参考方案4】:

这是一个非常常见的问题,有一个非常简单的解决方案:cronjoblock 一个简单的 8 行 shellscript 包装器使用flock 应用锁定:

https://gist.github.com/coderofsalvation/1102e56d3d4dcbb1e36f

顺便说一句。 cronjoblock 还可以逆转 cron 的垃圾邮件行为:仅在出现问题时才输出。这对于 cron 的 MAILTO 变量很方便。 stdout/stderr 输出将被抑制(因此 cron 不会发送邮件)除非给定进程的退出代码 > 0

【讨论】:

【参考方案5】:

flock 在 php 5.3.3 中将不起作用,因为文件资源句柄关闭时的自动解锁已被删除。现在解锁总是必须手动完成。

【讨论】:

【参考方案6】:

我用这个 ::

<?php
// Create a PID file
if (is_file (dirname ($_SERVER['SCRIPT_NAME']) . "/.processing"))  die (); 
file_put_contents (dirname ($_SERVER['SCRIPT_NAME']) . "/.processing", "processing");

// SCRIPT CONTENTS GOES HERE //

@unlink (dirname ($_SERVER['SCRIPT_NAME']) . "/.processing");
?>

【讨论】:

如果您的脚本因任何原因崩溃,cron 将永远不会再次运行。【参考方案7】:
#!/bin/bash
ps -ef | grep -v grep | grep capture_12hz_sampling_track.php
if [ $? -eq 1 ];
then
     nohup /usr/local/bin/php /opt/Apache/htdocs/cmsmusic_v2/script/Mp3DownloadProcessMp4/capture_12hz_sampling_track.php &
else
      echo "Already running"
fi

【讨论】:

【参考方案8】:

另一种选择:

<?php

/**
* Lock manager to ensure our cron doesn't run twice at the same time.
*
* Inspired by the lock mechanism in Mage_Index_Model_Process
*
* Usage:
* 
* $lock = Mage::getModel('stcore/cron_lock');
*
* if (!$lock->isLocked()) 
*      $lock->lock();
*      // Do your stuff
*      $lock->unlock();
* 
*/
class ST_Core_Model_Cron_Lock extends Varien_Object

    /**
     * Process lock properties
     */
    protected $_isLocked = null;
    protected $_lockFile = null;

    /**
     * Get lock file resource
     *
     * @return resource
     */
    protected function _getLockFile()
    
        if ($this->_lockFile === null) 
            $varDir = Mage::getConfig()->getVarDir('locks');
            $file = $varDir . DS . 'stcore_cron.lock';
            if (is_file($file)) 
                $this->_lockFile = fopen($file, 'w');
             else 
                $this->_lockFile = fopen($file, 'x');
            
            fwrite($this->_lockFile, date('r'));
        
        return $this->_lockFile;
    

    /**
     * Lock process without blocking.
     * This method allow protect multiple process runing and fast lock validation.
     *
     * @return Mage_Index_Model_Process
     */
    public function lock()
    
        $this->_isLocked = true;
        flock($this->_getLockFile(), LOCK_EX | LOCK_NB);
        return $this;
    

    /**
     * Lock and block process.
     * If new instance of the process will try validate locking state
     * script will wait until process will be unlocked
     *
     * @return Mage_Index_Model_Process
     */
    public function lockAndBlock()
    
        $this->_isLocked = true;
        flock($this->_getLockFile(), LOCK_EX);
        return $this;
    

    /**
     * Unlock process
     *
     * @return Mage_Index_Model_Process
     */
    public function unlock()
    
        $this->_isLocked = false;
        flock($this->_getLockFile(), LOCK_UN);
        return $this;
    

    /**
     * Check if process is locked
     *
     * @return bool
     */
    public function isLocked()
    
        if ($this->_isLocked !== null) 
            return $this->_isLocked;
         else 
            $fp = $this->_getLockFile();
            if (flock($fp, LOCK_EX | LOCK_NB)) 
                flock($fp, LOCK_UN);
                return false;
            
            return true;
        
    

    /**
     * Close file resource if it was opened
     */
    public function __destruct()
    
        if ($this->_lockFile) 
            fclose($this->_lockFile);
        
    

来源:https://gist.github.com/wcurtis/9539178

【讨论】:

【参考方案9】:

我正在运行一个 php cron 作业脚本,该脚本专门处理使用现有 API 发送文本消息。在我的本地机器上,cron 工作运行良好,但在我客户的机器上,它正在发送双重消息。虽然这对我来说没有意义,但我仔细检查了负责发送消息的文件夹的权限,并将权限设置为 root。一旦我将所有者设置为 www-data (Ubuntu),它就会开始正常运行。

这可能不是你的问题,但如果它是一个简单的 cron 脚本,我会仔细检查权限。

【讨论】:

以上是关于如何防止 cron 作业执行(如果它已经在运行)的主要内容,如果未能解决你的问题,请参考以下文章

如何指示 cron 每两周执行一次作业?

如何在Hasura上创建cron作业?

一个永远不会执行的 cron 作业

从 root 帐户执行 cron 作业是不是安全?

将 Cron 作业存储在哪里,它们会一直运行吗?

如何记录 cron 作业?