Идентификатор процесса проверки PHP
это то, что я подумал некоторое время и решил спросить об этом.
у нас есть функция getmypid (), которая вернет текущий идентификатор процесса скриптов. Есть ли какая-то функция, например
checkifpidexists () в php? Я имею в виду встроенное, а не какое-то пакетное решение скрипта.
и есть ли способ изменить PID скриптов?
некоторые разъяснения:
Я хочу проверить, существует ли pid, чтобы увидеть, является ли скрипт уже работает, поэтому он не запускается снова, faux cron job, если хотите.
причина, по которой я хотел изменить pid, заключается в том, что я могу установить PID скрипта на что-то действительно высокое, например 60000 и жесткий код, который имеет значение, поэтому этот скрипт может работать только на этом pid, поэтому будет работать только 1 экземпляр
правка----
чтобы помочь кому - либо еще с этой проблемой, я создал этот класс:
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';
}
6 ответов
в 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 );
идентификаторы процессов предоставляются ОС и нельзя зарезервировать идентификатор процесса. Вы бы написали свой демон, чтобы уважать PID-файл.
лучший способ сделать это-использовать 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";
}
?>
здесь не так много ошибок, но быстрый запуск показывает, что это работает в моей системе.
нет вы не можете изменить какие-либо процессы pid. Он назначается ядром и является частью структур данных ядра
для проверки наличия PID на машине windows я использую:
function pidExists($pid)
{
exec('TASKLIST /NH /FO "CSV" /FI "PID eq '.$pid.'"', $outputA );
$outputB = explode( '","', $outputA[0] );
return isset($outputB[1])?true:false;
}
отметим, что $outputb одновременно[0] содержит сообщения, которые pid не может быть найден, если pid действительно не существует! Поэтому для проверки я использую второй аргумент.
EDIT:
чтобы расширить это, его также можно динамически создавать скрипты в windows в фоновом режиме с помощью 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 чтобы установить заголовок этого процесса в какой-то уникальный хэш.
внутри родителя, породившего дочерний процесс, вы можете использовать следующий код, чтобы найти этот процесс в tasklist, используя его windowtitle для поиска uniquehash.
exec('TASKLIST /NH /FO "CSV" /FI "windowtitle eq ' . escapeshellarg($uniquehash) . '"', $output );
в сочетании с базой данных, вы можете по существу построить workermanager общение между различными PHP-скрипты.
как говорили другие, вы не можете изменить идентификатор процесса - он назначается и полностью управляется ядром ОС. Кроме того, вы не сказали, является ли это командной строкой или веб-сервером: если это последний, Вы даже не можете получить pid своего скрипта.
на страница руководства для getmypid() содержит некоторые примеры" оптимистической " блокировки. Я использую слово optimisitc, поскольку PHP никогда не будет приближаться к подобным asp.net веб-приложение, где у вас есть истинная потоковая среда с общими / статическими классами и, следовательно, Singleton для использования/злоупотребления. В основном у вас есть возможность:
- трогательно где-то "файл" на файл-системе. Затем ваш скрипт проверяет, существует ли этот файл: если он существует, завершите, в противном случае коснитесь этого файла и продолжите обработку
- установка флага на основе базы данных, чтобы сказать, что скрипт запущен. Как указано выше, но используйте таблицу/поле db, чтобы пометить скрипт как бегущий.
оба они полагаются на правильное завершение скрипта (так как последним шагом было бы удаление флага файла блокировки/db). Если сценарий аварийно завершает работу по какой-либо причине (или сама машина), вы можете остаться с ручным процессом очистки, чтобы удалить флаг. Для этого нет простого решения, но один из путей для изучения-это посмотреть на дату-штамповку замка с помощью арбитра "если старше X, последний запуск должен был разбиться".
Не забывайте также, что вы можете получить доступ к командам оболочки через backticks ( ` ), что даст вам доступ к стандартным инструментам *nix для работы с pids.
источник:http://www.php.net/manual/en/language.operators.execution.php