这是我想知道的一段时间,并决定询问它。
我们有函数getmypid(),它将返回当前脚本进程id。是否有某种功能,如
php中的checkifpidexists()?我的意思是内置的,而不是一些批处理脚本解决方案。有没有办法更改脚本pid?
一些澄清:
我想检查一个pid是否存在,看看脚本是否已经在运行,所以它不再运行,如果你愿意的话,还可以使用faux cron job。
我想改变pid的原因是我可以将脚本pid设置为真正高的东西,例如60000和硬编码值,这样这个脚本只能运行在那个pid上,所以只运行它的一个实例
EDIT ----
为了帮助其他人解决这个问题,我创建了这个课程:
class instance {
private $lock_file = '';
private $is_running = false;
public function __construct($id = __FILE__) {
$id = md5($id);
$this->lock_file = sys_get_temp_dir() . $id;
if (file_exists($this->lock_file)) {
$this->is_running = true;
} else {
$file = fopen($this->lock_file, 'w');
fclose($file);
}
}
public function __destruct() {
if (file_exists($this->lock_file) && !$this->is_running) {
unlink($this->lock_file);
}
}
public function is_running() {
return $this->is_running;
}
}
你可以这样使用它:
$instance = new instance('abcd'); // the argument is optional as it defaults to __FILE__
if ($instance->is_running()) {
echo 'file already running';
} else {
echo 'file not running';
}
答案 0 :(得分:17)
在linux中,你会看到/ proc。
return file_exists( "/proc/$pid" );
在Windows中你可以使用shell_exec()tasklist.exe,这会找到匹配的进程ID:
$processes = explode( "\n", shell_exec( "tasklist.exe" ));
foreach( $processes as $process )
{
if( strpos( "Image Name", $process ) === 0
|| strpos( "===", $process ) === 0 )
continue;
$matches = false;
preg_match( "/(.*?)\s+(\d+).*$/", $process, $matches );
$pid = $matches[ 2 ];
}
我相信你要做的是维护一个PID文件。在我编写的守护进程中,他们检查配置文件,查找pid文件的实例,从pid文件中获取pid,检查是否存在/ proc / $ pid,如果不存在,则删除pid文件。
if( file_exists("/tmp/daemon.pid"))
{
$pid = file_get_contents( "/tmp/daemon.pid" );
if( file_exists( "/proc/$pid" ))
{
error_log( "found a running instance, exiting.");
exit(1);
}
else
{
error_log( "previous process exited without cleaning pidfile, removing" );
unlink( "/tmp/daemon.pid" );
}
}
$h = fopen("/tmp/daemon.pid", 'w');
if( $h ) fwrite( $h, getmypid() );
fclose( $h );
进程ID由操作系统授予,并且无法保留进程ID。你可以编写你的守护进程来尊重pid文件。
答案 1 :(得分:6)
实现此目的的更好方法是使用pid或锁定文件。只需检查pid文件是否存在,根据需要创建它,并使用正在运行的pid填充它。
<?
class pidfile {
private $_file;
private $_running;
public function __construct($dir, $name) {
$this->_file = "$dir/$name.pid";
if (file_exists($this->_file)) {
$pid = trim(file_get_contents($this->_file));
if (posix_kill($pid, 0)) {
$this->_running = true;
}
}
if (! $this->_running) {
$pid = getmypid();
file_put_contents($this->_file, $pid);
}
}
public function __destruct() {
if ((! $this->_running) && file_exists($this->_file)) {
unlink($this->_file);
}
}
public function is_already_running() {
return $this->_running;
}
}
?>
使用如下:
<?
$pidfile = new pidfile('/tmp', 'myscript');
if($pidfile->is_already_running()) {
echo "Already running.\n";
exit;
} else {
echo "Started...\n";
}
?>
此处没有太多错误检查,但快速运行显示这在我的系统上有效。
答案 2 :(得分:1)
不,你不能改变任何进程pid。它由内核分配,是内核数据结构的一部分
答案 3 :(得分:1)
为了检查Windows机器上是否存在PID,我使用:
function pidExists($pid)
{
exec('TASKLIST /NH /FO "CSV" /FI "PID eq '.$pid.'"', $outputA );
$outputB = explode( '","', $outputA[0] );
return isset($outputB[1])?true:false;
}
请注意,如果pid确实不存在, $ outputB [0] 包含无法找到pid的消息!所以要验证我使用第二个参数。
修改强>
为了对此进行扩展,还可以使用PowerShell在后台动态生成脚本,如下所示:
// this function builds an argument list to parse into the newly spawned script.
// which can be accessed through the superglobal global $argv;
function buildArgList( array $arguments){
return ' '. implode(' ', $arguments) .' ';
}
$arguments = buildArgList(['argument1','argument2','argument3']);
$windowstyle = 'normal'; // or you can use hidden to hide the CLI
pclose(popen("powershell start-process -FilePath '" . PHP_BINARY . "' -ArgumentList '-f\"" . $file . " " . $arguments . "\"' -WindowStyle " . $windowstyle,"r"));
您生成的脚本可以使用:cli_set_process_title 将该进程的标题设置为某个唯一的哈希。
在生成子进程的父进程中,您可以使用以下代码在任务列表中使用其windowtitle搜索uniquehash来查找该进程。exec('TASKLIST /NH /FO "CSV" /FI "windowtitle eq ' . escapeshellarg($uniquehash) . '"', $output );
当与数据库结合使用时,您基本上可以构建一个workermanager 在不同的PHP脚本之间进行通信。
答案 4 :(得分:0)
正如其他人所说,你无法改变进程ID - 它由操作系统内核分配并完全管理。此外,您还没有说过这是基于命令行还是基于Web服务器:如果是后者,您甚至可能无法获得脚本的pid。
manual page for getmypid()包含一些“乐观”锁定的例子。我使用的词是optimisitc,因为PHP永远不会接近像asp.net web应用程序那样你有一个真正的带有共享/静态类的线程环境,因此使用/滥用Singleton。基本上你可以选择:
这两个都依赖于脚本正确终止(因为最后一步是删除锁文件/ db标志)。如果脚本因任何原因(或机器本身)崩溃,您可以通过手动整理过程来删除该标志。对此没有简单的解决方案,但是探索的一条途径是查看锁定日期标记,使用“如果年龄大于X,最后一次运行必须已崩溃”的方法。
答案 5 :(得分:0)
不要忘记您可以通过反引号(`)访问shell命令,这样您就可以访问标准的* nix工具来处理pids。
来源:http://www.php.net/manual/en/language.operators.execution.php