>  기사  >  데이터 베이스  >  PHP에서 데몬 프로세스를 구현하는 두 가지 일반적인 방법

PHP에서 데몬 프로세스를 구현하는 두 가지 일반적인 방법

黄舟
黄舟원래의
2017-09-08 15:12:481498검색

이 글에서는 주로 PHP 데몬 프로세스의 두 가지 일반적인 구현 방법을 소개합니다. PHP 데몬 프로세스의 원리, 관련 구현 방법 및 작동 주의 사항을 구체적인 예의 형태로 분석합니다. PHP 설명 데몬 프로세스에는 두 가지 일반적인 구현이 있습니다. 참고용으로 모두와 공유하세요. 자세한 내용은 다음과 같습니다.

첫 번째 방법은 nohup과 &를 함께 사용하는 것입니다. 명령 뒤에 앰퍼샌드를 추가하면 콘솔을 차지하지 않고 백그라운드에서 실행할 수 있습니다. 여기서는 while 무한 루프를 사용하여 보여줍니다.

<?php
while(true){
    echo time().PHP_EOL;
    sleep(3);
}

프로세스 시작을 위한 & 메소드


[root@localhost php]# php deadloop.php &
[1] 3454
[root@localhost php]# ps aux | grep 3454
root   3454 0.0 0.8 284544 8452 pts/0  T  18:06  0:00 php deadloop.php
root   3456 0.0 0.0 103316  896 pts/0  S+  18:08  0:00 grep 3454
[1]+ Stopped         php deadloop.php
[root@localhost php]#

프로세스가 콘솔을 점유하지 않는 것을 볼 수 있습니다. 이때 콘솔은 다른 명령을 실행할 수도 있습니다. 콘솔을 점유하는 일반 모드로 프로세스를 진행합니다.


[root@localhost php]# fg
php deadloop.php
1470996682
1470996685
1470996688
1470996691

위는 & 명령에 대한 간략한 소개입니다

다른 명령 nohup을 살펴보겠습니다

명령 앞에 nohup을 추가하면 시작된 프로세스는 Linux 정지 신호(SIGHUP)를 무시하고 어떤 상황에서 트리거됩니다. Linux에서 SIGHUP 신호는? 다음 내용은 Baidu Encyclopedia에서 가져온 것입니다.

SIGHUP은 다음 3가지 상황에서 해당 프로세스로 전송됩니다.

1. 터미널이 닫히면 세션으로 신호가 전송됩니다. 첫 번째 프로세스와 작업으로 제출된 A 프로세스(즉, & 기호와 함께 제출된 프로세스)

2. 세션의 첫 번째 프로세스가 종료되면 세션의 포그라운드 프로세스 그룹에 있는 각 프로세스에 신호가 전송됩니다. 3 상위 프로세스가 종료되면 프로세스 구성은 고아(orphan) 프로세스 그룹이 되며, 프로세스 그룹의 프로세스가 정지(SIGSTOP 또는 SIGTSTP 신호 수신)되면 이 신호는 프로세스 그룹의 모든 프로세스로 전송됩니다.

1과 2를 결합하면 프로세스가 &(작업 모드)에서 시작되었는지 여부에 관계없이 터미널을 닫을 때 SIGHUP 신호가 수신된다는 것을 알 수 있습니다. 그러면 프로세스가 SIGHUP 신호를 수신하면 어떻게 될까요? Baidu Encyclopedia에서 가져온 동일한 문장을 참조하세요.


시스템의 SIGHUP 신호 처리 기본은 신호를 수신하는 프로세스를 종료하는 것입니다. 따라서 프로그램에서 신호가 캡처되지 않으면 신호가 수신될 때 프로세스가 종료됩니다.

즉, 터미널 프로세스를 닫으면 SIGHUP 신호가 수신되며, 이 신호의 기본 처리 방법은 프로세스를 종료하는 것입니다. 물론 우리가 직접 신호를 처리하거나 무시할 수도 있습니다. 위 루프의 예를 살펴보겠습니다. 약간의 개선 사항을 추가해 보겠습니다.

<?php
declare(ticks = 1);
pcntl_signal(SIGHUP, function(){
    // 这地方处理信号的方式我们只是简单的写入一句日志到文件中
    file_put_contents(&#39;logs.txt&#39;, &#39;pid : &#39; . posix_getpid() . &#39; receive SIGHUP 信号&#39; . PHP_EOL);
});
while(true){
    echo time().PHP_EOL;
    sleep(3);
}

그렇게 번거로울 필요는 없습니다. 하지만 nohup을 사용하여 프로세스를 시작하고 종료할 때는 그렇습니다. 터미널에서 프로세스는 SIGHUP 신호를 무시하고 종료되지 않습니다. 우선 지금 신호 처리 코드를 제거하십시오. 그런 다음 nohup을 실행합니다.


[root@localhost php]# nohup php deadloop.php

nohup: 입력을 무시하고 출력을 "nohup.out"에 추가합니다

그리고 nohup은 기본적으로 프로그램의 출력을 현재 디렉터리의 nohup.out 파일로 리디렉션합니다. write $homepath/nohup.out

[root@localhost php]# ls
cmd.sh deadloop.php getPhoto.php nohup.out pics
[root@localhost php]# tail -f nohup.out
1470999772
1470999775
1470999778
1470999781
1470999784
1470999787
1470999790
1470999793
1470999796
1470999799
1470999802

이때 터미널을 닫으세요. 프로세스는 종료되지 않지만 프로세스를 생성한 상위 프로세스가 종료되었기 때문에 고아 프로세스(ppid=1)가 됩니다.


[root@localhost ~]# ps -ef | grep 3554
root   3554 3497 0 19:09 pts/0  00:00:00 php deadloop.php
root   3575 3557 0 19:10 pts/1  00:00:00 grep 3554
[root@localhost ~]# ps -ef | grep 3554
root   3554   1 0 19:09 ?    00:00:00 php deadloop.php
root   3577 3557 0 19:10 pts/1  00:00:00 grep 3554
[root@localhost ~]#

결론:

nohup과 & 메소드를 결합하면 시작된 프로세스는 콘솔을 차지하지 않고 콘솔에 의존하지 않으며 콘솔이 닫힌 후에는 프로세스 No. 1이며 고아 프로세스가 되며 이는 데몬 프로세스의 메커니즘과 매우 유사합니다.

[root@localhost php]# nohup php deadloop.php >logs.txt 2>error.txt &
[1] 3612
[root@localhost php]# ps -ef |grep 3612
root   3612 3557 0 19:18 pts/1  00:00:00 php deadloop.php
root   3617 3557 0 19:19 pts/1  00:00:00 grep 3612
[root@localhost php]#
>logs.txt가 표준 출력을 리디렉션하는 경우 2>error.txt는 표준 오류 출력을 리디렉션합니다.

위는 첫 번째 구현 방법에 대한 소개입니다.

두 번째 구현 방법은 데몬 프로세스의 규칙과 특성에 따라 코드를 통해 구현하는 것입니다

. 데몬 프로세스의 가장 큰 특징은 사용자 단말과 세션이 분리되어 있다는 점입니다

. 구현된 코드(주요 위치에 주석 포함)

<?php
$pid = pcntl_fork();
if ($pid == -1)
{
  throw new Exception(&#39;fork子进程失败&#39;);
}
elseif ($pid > 0)
{
  //父进程退出,子进程变成孤儿进程被1号进程收养,进程脱离终端
  exit(0);
}
// 最重要的一步,让该进程脱离之前的会话,终端,进程组的控制
posix_setsid();
// 修改当前进程的工作目录,由于子进程会继承父进程的工作目录,修改工作目录以释放对父进程工作目录的占用。
chdir(&#39;/&#39;);
/*
 * 通过上一步,我们创建了一个新的会话组长,进程组长,且脱离了终端,但是会话组长可以申请重新打开一个终端,为了避免
 * 这种情况,我们再次创建一个子进程,并退出当前进程,这样运行的进程就不再是会话组长。
 */
$pid = pcntl_fork();
if ($pid == -1)
{
  throw new Exception(&#39;fork子进程失败&#39;);
}
elseif ($pid > 0)
{
  // 再一次退出父进程,子进程成为最终的守护进程
  exit(0);
}
// 由于守护进程用不到标准输入输出,关闭标准输入,输出,错误输出描述符
fclose(STDIN);
fclose(STDOUT);
fclose(STDERR);
/*
 * 处理业务代码
 */
while(TRUE)
{
  file_put_contents(&#39;log.txt&#39;, time().PHP_EOL, FILE_APPEND);
  sleep(5);
}
그게 다예요!

위 내용은 PHP에서 데몬 프로세스를 구현하는 두 가지 일반적인 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.