PHP が Web 開発でタイマーを使用できない根本的な理由は、制御可能な常駐メモリ操作環境が欠如していることです。2 つの重要な点、1 つ目は常駐メモリ、2 つ目は制御可能です。 CGI モードでは、プロセスはスクリプトの実行後に直接終了し、指定された時間にタスクを実行することは期待できません。PHP-FPM モードでは、プロセスはメモリ内に常駐しますが、制御できません。
このチュートリアルの動作環境: Windows7 システム、PHP8 バージョン、DELL G3 コンピューター
一般的なタイマーは 2 つあり、1 つは定期的に実行される (たとえば、毎日午前 3 時にレポートが発行される)、もう 1 つは指定された時間後に (1 回だけ) 実行されます。 , たとえば、各メンバーはシステムにログインしてから 5 分後に発行されます。毎日のログイン特典。 JavaScriptのsetInterval関数やsetTimeout関数に相当します(厳密にはsetIntervalは定期的に実行され、指定された時点での実行は自身で処理する必要があります)。
Web 開発を行う PHP プログラマは、JavaScript の 2 つのタイマー関数をよく知っているはずです。PHP レベルに戻ると少しびっくりします:
PHP にはスリープがありますが、いいえ(内蔵)タイマー機能が利用可能です。 スリープ機能はかろうじてそれを実行できますが、プロセスがブロックされ、この期間中は他のことができなくなります (または応答しなくなります)。なぜ PHP はタイマー機能を提供しないのですか?
理由
PHP が Web 開発でタイマーを使用できない根本的な理由は、制御可能な常駐メモリ操作環境が欠如していることです。 2 つの重要なポイント: 1 つ目は常駐メモリ、2 つ目は制御可能です。 CGI モードでは、プロセスはスクリプトの実行後に直接終了し、指定された時間にタスクを実行することは期待できません。PHP-FPM モードでは、プロセスは (ほとんど) メモリ内に常駐しますが、制御できません。
制御不能とは、PHP を実行するプロセスが PHP コードの影響を受けず、プロセスのエントリ ポイントと終了タイミングが追加のプログラムによって制御されることを意味します。たとえば、FPM モードでは、PHP スクリプトの exit 関数と die 関数はスクリプトの実行を中断するだけで、スクリプトの実行プロセスに特別な影響を与えません (メモリ リークを除く)。 PHP 開発者が作成したスクリプトはプロセスの実行本体となり、実行後はプロセスの実行コンテキストからアンロードされます。この場合、PHP スクリプトの実行タイミングは依然として外部によって決定され、外部からの要求がない場合、PHP コードは何もせずにハードディスク上に静かに存在し、スケジュールされたタスクになります。
PHP は主に Web 開発を目的としているため、PHP の実行モードは安定していて信頼性が高く、開発効率が高速です。たとえば、リソースのリリース手順を省略すると、開発における多くの作業負荷や落とし穴を回避できます。タイムゾーンや文字エンコーディングなどを変更しても復元しないサードパーティのライブラリ コードについて考えてみると、常駐メモリを実行している環境では、ほぼ確実に後続のリクエストで問題が発生します。ただし、FPM モードでは、この落とし穴が意図せず解消され、デバッグ時間が大幅に節約され、プログラマがヘアラインを維持する能力に大きく貢献します。
問題は理解されました。では、PHP でスケジュールされたタスクを実行するためにタイマーを使用するにはどうすればよいでしょうか?
Web環境では、PHPスクリプトにはデフォルトでタイムアウトが設定されています。タイムアウト設定を削除すると、プログラムをバックグラウンドで実行し続けることができます (プロセスが終了しない場合)。たとえば、次のコードはリクエストに応答した後もバックグラウンドで実行を続け、5 秒ごとに時刻をファイルに出力します。
# test.php set_time_limit(0); # 取消超时设置,让脚本可一直运行 echo 'This is a background run forever script. Now you can leave me alone.'; fastcgi_finish_request(); # 结束当前请求 do{ file_put_contents("/tmp/out.dat", "test script, now:" . date("Y-m-d H:i:s") . "\n", FILE_APPEND); sleep(5); }while(true);
http://localhost:8080/ をリクエストした後test.php
ファイル、/tmp/out.dat
ファイルを監視すると、クライアントが切断するか、ブラウザを閉じるか、コンピュータを再起動するかに関係なく、コンテンツが常に出力されていることがわかります (サーバーを再起動できません)。これは、プログラムが実行され、必要なタイマー機能が実現されたことを示しています。 sleep
を usleep
、time_nanosleep
に変更すると、マイクロ秒およびナノ秒レベルのタイマーも実装できると思いませんか?
実際には、この方法でタイマーを実装することは避けるべきです。これは非効率であるだけでなく、若干の危険があるためです。理由の 1 つは、各リクエストが 1 つのプロセスを占有し、10 万のリクエストには 10 万のプロセスが必要となるため、基本的にシステムがクラッシュしたり、後続のリクエストが応答しなくなったりすることです。また、セッションは開いているが、 session_write_close
を呼び出すのを忘れると、同じユーザーからの後続のリクエストがハングします (セッションはアクティブなときはロック状態にあり、セッションを閉じることに失敗すると、後続のプロセスを開けなくなります)セッション)。
Web 開発はユーザーのリクエストにできるだけ早く応答する必要があります。このように Web 開発でタイマーを強制すると、Web アプリケーション全体が不安定になり、信頼性が低くなり、予測不能になります。孟子はこう言いました、「知って慎重に行動せよ、君子は危険な壁の下に立つものではない」信頼できない慣行は可能な限り避けるべきであり、責任転嫁や責任転嫁も避けるべきです。
次に、PHP でタイマーを使用するための正しい姿勢を見てみましょう。
PHP でタイマーを実装する方法を簡単にまとめると、次のようになります。
cron を使用し、スケジュールを設定するJenkins などのツールは、定期的にスケジュールされたタスク (スクリプトの実行または特定の URL のリクエスト) を実行します。
1 回限りの実行タスクは、メッセージ キュー、データベース、など。サードパーティ プログラムの実行;
3 番目の方法を除く他の方法はすべて推奨されますが、実際のニーズに基づいて具体的なプランを検討してください。 PHP プログラマーとしては、もちろん、CLI モードである PHP を使用することが第一の選択です。
正直に言うと、CLI モードを使用すると、PHP の領域を大幅に拡張できます。 CLI モードでは、プログラムのエントリ ポイントはスクリプトであり、コードはメモリ内に常駐することができ、プロセスは PHP コードによって完全に制御されます。この形式では、タイマーを実装するさまざまな方法があります。この記事では、他の人にインスピレーションを与えるためのいくつかの方法をリストします。
組み込み (高精度) を備えた swoole
、workerman
などのフレームワークを使用するタイマー;
マルチプロセス (プール)/マルチスレッド (プール) テクノロジを使用します (pcntl
、pthreads
拡張は、 CLI モード);
ティックやアラームなどの信号を処理する;
libevent
などのイベント駆動型ライブラリを使用するおよび libev
;
sleep
ループを追加するか、イベント ループを自分で実装します。
いじりたい場合は、2-5 プランを自分で使用してください。swoole
、workerman# をいじりたくない場合は、 ## および他のフレームワークでは、安定性と信頼性が高く、第一の選択肢となります。
PHP ビデオ チュートリアル 」
以上がなぜPHPにはタイミング関数がないのでしょうか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。