検索
ホームページバックエンド開発PHPチュートリアルPHP7におけるデーモン実装の詳細説明

この記事では主に PHP7 を使用してデーモン デーモン プロセスを実装する方法について説明します。一定の参考価値があるので、興味のある方は学習してください。

マルチタスク コンピュータ オペレーティング システムでは、デーモン プロセス はバックグラウンドで実行されるコンピュータ プログラムです。このようなプログラムはプロセスとして初期化されます。デーモン プログラムの名前は通常、文字「d」で終わります。たとえば、syslogd はシステム ログを管理するデーモンを指します。

デーモン プログラムは、常時実行されているサーバー プログラムであり、デーモン プロセスとも呼ばれます。通常、デーモン プログラムはシステムのバックグラウンドで実行されます。制御端末はなく、フォアグラウンドとは対話しません。通常、デーモン プログラムはシステム サービスとして使用されます。デーモンは、通常、システムの起動後に実行され、システムのシャットダウン時に終了する、長時間実行されるプロセスです。一般に、デーモン プログラムは制御端末を持たず、フォアグラウンドでユーザーと対話できないため、バックグラウンドで実行されます。デーモン プログラムは通常、クライアント プログラムが通信するのを待つサービス プログラムとして使用されます。実行中のデーモン プログラムをデーモン プロセスとも呼びます。

通常、デーモン プロセスには既存の親プロセス (つまり、PPID=1) がなく、UNIX システム プロセス階層の init の直下に配置されます。デーモン プロセスは通常、次の方法で自分自身をデーモンにします。 子プロセスで fork を実行し、その親プロセスをすぐに終了して、子プロセスを ## で使用できるようにします。 #init で実行します。この方法は、「シェル処理」と呼ばれることがよくあります。

システムは通常、起動時にデーモン プロセスを開始します。デーモンは、特定のタスクを通じてネットワーク要求、ハードウェアアクティビティ、または他のアプリケーションからのその他の要求に応答するためのサポートを提供します。デーモンは、ハードウェア (一部の Linux システムの devfsd など) を構成したり、スケジュールされたタスク (cron など) を実行したり、その他のタスクを実行したりすることもできます。各プロセスには親プロセスがあり、子プロセスが終了すると、親プロセスは子プロセスの終了ステータスを取得できます。

デーモン プロセスは、単にターミナルを使用せずにバックグラウンドで実行できるプロセスです。これは Linux では非常に一般的なプロセスです。たとえば、Apache や mysql などのサービスが開始された後などです。 , デーモンプロセスとしてメモリ上に常駐します。デーモンはバックグラウンドで実行されるアプリケーションであり、ユーザーが直接操作することはありません。デーモンの例としては、Cron や MySQL があります。 PHP デーモンの使用は非常に簡単で、PHP 4.1 以降のコンパイル パラメータが必要です。 #--enable-pcntl

# #バックグラウンドで実行する必要がある時間のかかるタスクがあるとします: キャッシュを予熱するためにすべての mysql のユーザー テーブル内の 2,000 万ユーザーすべてを Redis にインポートすると、このタスクはおそらくしばらく終了しないでしょう。 、システム内でデーモンとして実行する php スクリプトを記述する必要があり、終了後に自動的に起動されます。

Linux では、バックグラウンド スクリプトには 3 つの方法があります:

1. コマンド # の後に & 記号 ## を追加します。 例: php task.php &. この方法の欠点は、ターミナルが正常に閉じられたか異常に閉じられたかに関係なく、ターミナルが閉じられた場合、php プロセスが終了したときに php プロセスも閉じられることです。次に、コード内に echo や print_r などの出力テキストがある場合は、現在のターミナル ウィンドウに出力されます。

2. nohup コマンド

を使用します。たとえば、

nohup php task.php &

。デフォルトでは、echoまたは、print_r で出力したテキストが PHP コードと同じディレクトリにある nohup.out ファイルに出力されます exit コマンドや閉じるボタンなど通常の方法でターミナルを閉じた場合、プロセスは実行されません閉じられており、バックグラウンドで引き続き実行されます。ただし、ターミナルが異常な終了または終了に遭遇した場合、php プロセスもすぐに終了します。本質的に、これは安定した信頼性の高いデーモン ソリューションではありません。 ##3.

pcntl

および posix 拡張機能を介して実装されますプログラミング中に注意すべき点は次のとおりです:

  • セカンダリ pcntl_fork() および posix_setsid を介してメイン プロセスを端末から離れるようにします
  • ## を介して# pcntl_signal() 無視または処理 SIGHUP signal
  • マルチプロセス プログラムは 2 回渡す必要があります pcntl_fork() または pcntl_signal () SIGCHLD シグナルを無視して、子プロセスがゾンビ プロセスになるのを防ぎます
  • umask( )、継承されたファイル権限からの権限の影響を防止します
  • 実行中のプロセスの STDIN/STDOUT/STDERR/dev/ にリダイレクトします。 null または他のストリーム上で

##デーモンには次の特性があります:

    いいえターミナル
  • バックグラウンド実行
  • 親プロセスの PID は 1
必要です実行中のデーモンの表示 プロセスは、

ps -ax または ps -ef を介して表示できます。-x は、端末を制御しないプロセスがリストされることを意味します。

fork システム コール

fork システム コールは、親プロセスとほぼ同一のプロセスをコピーするために使用されます。プロセスが異なります。問題は、親プロセスの PID とメモリ空間が異なることです。コード ロジックに従って、親プロセスと子プロセスは同じ作業を完了することも、異なる作業を完了することもできます。子プロセスは、ファイル記述子などのリソースを親プロセスから継承します。

PHP の

pcntl 拡張機能は、PHP で新しいプロセスをフォークするために使用される pcntl_fork() 関数を実装します。

setsid システム コール

setsid システム コールは、新しいセッションを作成し、プロセス グループ ID を設定するために使用されます。ここにはいくつかの概念があります:

セッションプロセス グループ

Linux では、ユーザーのログインによりセッションが生成され、セッションには 1 つ以上のプロセス グループが含まれ、プロセス グループには複数のプロセスが含まれます。各プロセス グループにはグループ リーダー (

セッション リーダー) があり、その pid はプロセス グループのグループ ID です。プロセス リーダーがターミナルを開くと、このターミナルは制御ターミナルと呼ばれます。制御端末で例外(切断、ハードウェアエラーなど)が発生すると、プロセスグループリーダーに信号が送信されます。

バックグラウンドで実行されているプログラム (

& で終わるシェル実行命令など) も、端末が閉じられた後、つまり制御端末が切断されたときに強制終了されます。 SIGHUP シグナルが発行され、SIGHUP シグナルのデフォルトの動作はプロセスを終了することです。 。

Call setsid システム コールの後、現在のプロセスは新しいプロセス グループを作成します。現在のプロセスでターミナルが開かれていない場合は、これは、プロセス グループ内に制御端末は存在しないため、端末を閉じてプロセスを強制終了しても問題はありません。

PHP の

posix 拡張機能は、PHP グループに新しいプロセスを設定するために使用される posix_setsid() 関数を実装します。

2 番目のフォークの役割

まず、

setsid システム コールはプロセスから呼び出すことができません。グループリーダーは -1 を返します。

2 番目のフォーク操作のサンプル コードは次のとおりです:

<span style="font-size: 16px;">$pid1 = pcntl_fork();

if ($pid1 > 0) {</span><br/><span style="font-size: 16px;">    // 父进程会得到子进程号,所以这里是父进程执行的逻辑
    exit(&#39;parent process. 1&#39;."\n");
} else if ($pid1 < 0) {
    exit("Failed to fork 1\n");
}

if (-1 == posix_setsid()) {
    exit("Failed to setsid\n");
}

$pid2 = pcntl_fork();

if ($pid2 > 0) {
    exit(&#39;parent process. 2&#39;."\n");
} else if ($pid2 < 0) {
    exit("Failed to fork 2\n");
}</span>

pcntl_fork() 関数は子プロセスを作成します。子プロセス PID (プロセス ID) と PPID (親プロセス ID) のみが親プロセスと異なります。

戻り値

成功すると、生成された子プロセスの PID が親プロセスの実行スレッドに返されます。子プロセスで実行されます。スレッドには

0 が返されます。 が失敗すると、親プロセスのコンテキストには -1 が返されます。子プロセスは作成されず、PHP が実行されます。エラーが発生します。

假定我们在终端中执行应用程序,进程为 a,第一次 fork 会生成子进程 b,如果 fork 成功,父进程 a 退出。b 作为孤儿进程,被 init 进程托管。

此时,进程 b 处于进程组 a 中,进程 b 调用 posix_setsid 要求生成新的进程组,调用成功后当前进程组变为 b。


php fork2.php 
parent process. 1
parent process. 2

此时进程 b 事实上已经脱离任何的控制终端,例程:


cli_set_process_title(&#39;process_a&#39;);

$pidA = pcntl_fork();

if ($pidA > 0) {
    exit(0);
} else if ($pidA < 0) {
    exit(1);
}

cli_set_process_title(&#39;process_b&#39;);

if (-1 === posix_setsid()) {
    exit(2);
}

while(true) {
    sleep(1);
}

执行程序之后:  


$ php cli-title.php 
$ ps ax | grep -v grep | grep -E &#39;process_|PID&#39;
  PID TTY      STAT   TIME COMMAND
15725 ?        Ss     0:00 process_b

重新打开一个shell窗口,效果一样,都在呢

从 ps 的结果来看,process_b 的 TTY 已经变成了 ,即没有对应的控制终端。

代码走到这里,似乎已经完成了功能,关闭终端之后 process_b 也没有被杀死,但是为什么还要进行第二次 fork 操作呢?

StackOverflow 上的一个回答写的很好:

The second fork(2) is there to ensure that the new process is not a session leader, so it won’t be able to (accidentally) allocate a controlling terminal, since daemons are not supposed to ever have a controlling terminal.

这是为了防止实际的工作的进程主动关联或者意外关联控制终端,再次 fork 之后生成的新进程由于不是进程组组长,是不能申请关联控制终端的。

综上,二次 fork 与 setsid 的作用是生成新的进程组,防止工作进程关联控制终端。 

写一个demo测试下


<?php
// 第一次fork系统调用
$pid_A = pcntl_fork();

// 父进程 和 子进程 都会执行下面代码
if ($pid_A < 0) {
    // 错误处理: 创建子进程失败时返回-1.
    exit(&#39;A fork error &#39;);
} else if ($pid_A > 0) {
     // 父进程会得到子进程号,所以这里是父进程执行的逻辑
    exit("A parent process exit \n");
}

// B 作为孤儿进程,被 init 进程托管,此时,进程 B 处于进程组 A 中

// 子进程得到的$pid为0, 所以以下是子进程执行的逻辑,受控制终端的影响,控制终端关闭则这里也会退出

// [子进程] 控制终端未关闭前,将当前子进程提升会会话组组长,及进程组的leader
// 进程 B 调用 posix_setsid 要求生成新的进程组,调用成功后当前进程组变为 B
if (-1 == posix_setsid()) {
    exit("Failed to setsid\n");
}

// 此时进程 B 已经脱离任何的控制终端

// [子进程]  这时候在【进程组B】中,重新fork系统调用(二次fork)
$pid_B = pcntl_fork();
if ($pid_B < 0) {
    exit(&#39;B fork error &#39;);
} else if ($pid_B > 0) {
    exit("B parent process exit \n");
}

// [新子进程] 这里是新生成的进程组,不受控制终端的影响,写写自己的业务逻辑代码
for ($i = 1; $i <= 100; $i++) {
    sleep(1);
    file_put_contents(&#39;daemon.log&#39;,$i . "--" . date("Y-m-d H:i:s", time()) . "\n",FILE_APPEND);
}

Window 下跑回直接抛出异常


php runtime\daemon.php
PHP Fatal error:  Uncaught Error: Call to undefined function pcntl_fork() in D:\phpStudy\PHPTutorial\WWW\notes\runtime\daemon.php:13
Stack trace:
#0 {main}
  thrown in D:\phpStudy\PHPTutorial\WWW\notes\runtime\daemon.php on line 13

Linux 下执行,输出结果


<span style="font-size: 16px;">php daemon.php</span><br/><span style="font-size: 16px;">...
97--2018-09-07 03:50:09
98--2018-09-07 03:50:10
99--2018-09-07 03:50:11
100--2018-09-07 03:50:12</span>

所以,现在即使关闭了终端,改脚本任然在后台守护进程运行

相关教程:PHP视频教程

以上がPHP7におけるデーモン実装の詳細説明の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明
この記事は博客园で複製されています。侵害がある場合は、admin@php.cn までご連絡ください。
php7检测tcp端口不好用怎么解决php7检测tcp端口不好用怎么解决Mar 22, 2023 am 09:30 AM

在php5中,我们可以使用fsockopen()函数来检测TCP端口。这个函数可以用来打开一个网络连接和进行一些网络通信。但是在php7中,fsockopen()函数可能会遇到一些问题,例如无法打开端口、无法连接到服务器等。为了解决这个问题,我们可以使用socket_create()函数和socket_connect()函数来检测TCP端口。

php7.0怎么安装mongo扩展php7.0怎么安装mongo扩展Nov 21, 2022 am 10:25 AM

php7.0安装mongo扩展的方法:1、创建mongodb用户组和用户;2、下载mongodb源码包,并将源码包放到“/usr/local/src/”目录下;3、进入“src/”目录;4、解压源码包;5、创建mongodb文件目录;6、将文件复制到“mongodb/”目录;7、创建mongodb配置文件并修改配置即可。

php7.0安装了插件还是显示未安装怎么办php7.0安装了插件还是显示未安装怎么办Apr 02, 2024 pm 07:39 PM

解决 PHP 7.0 中插件未显示已安装问题的方法:检查插件配置并启用插件。重新启动 PHP 以应用配置更改。检查插件文件权限,确保其正确。安装丢失的依赖项,以确保插件正常运行。如果其他步骤均失败,则重建 PHP。其他可能原因包括插件版本不兼容、加载错误版本或 PHP 配置问题。

php8和php7哪个好php8和php7哪个好Nov 16, 2023 pm 03:09 PM

PHP8相较于PHP7在性能、新特性和语法改进、类型系统、错误处理和扩展等方面都有一些优势和改进。然而,选择使用哪个版本要根据具体的需求和项目情况来决定。详细介绍:1、性能提升,PHP8引入了Just-in-Time(JIT)编译器,可以提高代码的执行速度;2、新特性和语法改进,PHP8支持命名参数和可选参数的声明,使得函数调用更加灵活;引入了匿名类、属性的类型声明等等。

php7.0怎么安装部署php7.0怎么安装部署Nov 30, 2022 am 09:56 AM

php7.0安装部署的方法:1、到PHP官网下载与本机系统对应的安装版本;2、将下载的zip文件解压到指定目录;3、打开命令行窗口,在“E:\php7”目录下运行“php -v”命令即可。

PHP 服务器环境常见问题指南:快速解决常见难题PHP 服务器环境常见问题指南:快速解决常见难题Apr 09, 2024 pm 01:33 PM

PHP服务器环境常见的解决方法包括:确保已安装正确的PHP版本和已复制相关文件到模块目录。临时或永久禁用SELinux。检查并配置PHP.ini,确保已添加必要的扩展和进行正确设置。启动或重启PHP-FPM服务。检查DNS设置是否存在解析问题。

php7怎么下载与安装(教程分享)php7怎么下载与安装(教程分享)Mar 23, 2023 pm 02:11 PM

随着互联网技术的发展,计算机编程语言也随之不断发展和更新。PHP作为一种广泛应用于Web开发领域的编程语言,在多年的发展中经历了多个版本的更新,而最新版的PHP7又在性能和稳定性上有了巨大提升。为了能更好地应用PHP编程语言,这篇文章将介绍PHP7的下载和安装教程,供初学者参考。

记录一次用strace诊断php占用系统资源过高的问题记录一次用strace诊断php占用系统资源过高的问题May 03, 2024 pm 04:31 PM

本地环境:redhat6.7系统。nginx1.12.1,php7.1.0,代码使用yii2框架问题:本地的web站需要用到elasticsearch服务。当php使用本地服务器搭建的elasticsearch时,本地的负载都是正常。当我使用aws的elasticsearchservice服务时,本地服务器出现负载经常过高的情况。查看nginx和php日志,发现没有异常。系统的并发连接数也不高。这时候想到我们老大给我讲的一个strace诊断工具。调试过程:查找一个php的子进程idstrace-

See all articles

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

mPDF

mPDF

mPDF は、UTF-8 でエンコードされた HTML から PDF ファイルを生成できる PHP ライブラリです。オリジナルの作者である Ian Back は、Web サイトから「オンザフライ」で PDF ファイルを出力し、さまざまな言語を処理するために mPDF を作成しました。 HTML2FPDF などのオリジナルのスクリプトよりも遅く、Unicode フォントを使用すると生成されるファイルが大きくなりますが、CSS スタイルなどをサポートし、多くの機能強化が施されています。 RTL (アラビア語とヘブライ語) や CJK (中国語、日本語、韓国語) を含むほぼすべての言語をサポートします。ネストされたブロックレベル要素 (P、DIV など) をサポートします。

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

EditPlus 中国語クラック版

EditPlus 中国語クラック版

サイズが小さく、構文の強調表示、コード プロンプト機能はサポートされていません

VSCode Windows 64 ビットのダウンロード

VSCode Windows 64 ビットのダウンロード

Microsoft によって発売された無料で強力な IDE エディター