确保Linux上只有一个正在运行的PHP进程

Ole*_*leg 3 php linux cron

我有一个 php 脚本每分钟从 cron 运行一次。但有时它的工作时间会超过1分钟。我的问题是:确保现在只有一个进程正在运行的最佳方法是什么?

我使用这段代码:

$output = shell_exec('ps aux | grep some_script.php | grep -v grep');   //get all processes containing "some_script.php" and exclude current grep process
$trimmed = rtrim($output, PHP_EOL); //trim newline symbol in the end
$processes = explode(PHP_EOL, $trimmed);    //get the array of lines (i.e. processes)
$procCnt = count($processes);   //get number of lines
if ($procCnt > 2) {
  echo "busy\n";
  exit();       //exit if number of processes more than 2 (see explaination below)
}
Run Code Online (Sandbox Code Playgroud)

如果有一个 some_script 进程从 cron 运行,shell_exec 将返回如下内容:

apache     13593  0.0  0.0   9228  1068 ?        Ss   18:20   0:00 /bin/bash -c php -f /srv/www/robot/some_script.php 2>&1
apache     13602  0.0  0.0 290640 10544 ?        S    18:20   0:00 php -f /srv/www/robot/some_script.php
Run Code Online (Sandbox Code Playgroud)

所以如果我的输出超过 2 行,我会调用 exit()

我想问:我走的路对吗?或者,还有更好的方法?

任何帮助,将不胜感激

Izk*_*ata 5

检查这种方式会创建一个竞争条件,其中两个进程可以同时检索列表,然后都决定退出。根据您尝试执行的操作,这可能是问题,也可能不是问题。

一个可能更好的选择是创建某种类型的锁。我使用的一个简单目录是仅在进程运行时存在的目录 -mkdir是原子的,它要么成功(没有其他进程正在运行),要么失败(另一个进程已经创建了它)。只需确保完成后将其删除:

if (!mkdir("lock_dir")) {
   echo "busy\n";
   exit();
}
register_shutdown_function(function() {
   rmdir("lock_dir");
});
Run Code Online (Sandbox Code Playgroud)

或者更好的是,它看起来像是flock为了类似的目的而制作的。这是手册中的示例:

<?php

$fp = fopen("/tmp/lock.txt", "r+");

if (flock($fp, LOCK_EX)) {  // acquire an exclusive lock
    ftruncate($fp, 0);      // truncate file
    fwrite($fp, "Write something here\n");
    fflush($fp);            // flush output before releasing the lock
    flock($fp, LOCK_UN);    // release the lock
} else {
    echo "Couldn't get the lock!";
}

fclose($fp);

?>
Run Code Online (Sandbox Code Playgroud)

只需锁定脚本的运行时,类似于我的第一个示例。