ホームページ >バックエンド開発 >PHPチュートリアル >PHP のスケジュールされたタスクと継続的プロセス (fsockopen) の例
Web サーバーは PHP スクリプトを実行しますが、実行結果が返されるまでに時間がかかる場合があり、後続のスクリプトは実行を続行するまでに長時間待機する必要があります。 時間のかかるスクリプトの実行結果を待たずに単純に実行をトリガーし、次の操作を直接実行したい場合は、fskokopen 関数を使用できます。
PHP はソケット プログラミングをサポートしており、fscokopen 関数はリモート ホスト接続へのハンドルを返し、fopen から返されたハンドルを使用するのと同じように、fwrite、fgets、fread などの操作を実行できます。 fsockopen を使用してローカル サーバーに接続し、スクリプトの実行をトリガーし、スクリプトの実行が完了するのを待たずにすぐに戻ることで、非同期 PHP 実行の効果が得られます。 [推奨: PHP ビデオチュートリアル]
例:
<? function triggerRequest($url, $post_data = array(), $cookie = array()){ $method = "GET"; //通过POST或者GET传递一些参数给要触发的脚本 $url_array = parse_url($url); //获取URL信息 $port = isset($url_array['port'])? $url_array['port'] : 80; $fp = fsockopen($url_array['host'], $port, $errno, $errstr, 30); if (!$fp) { return FALSE; } $getPath = $url_array['path'] ."?". $url_array['query']; if(!empty($post_data)){ $method = "POST"; } $header = $method . " " . $getPath; $header .= " HTTP/1.1\r\n"; $header .= "Host: ". $url_array['host'] . "\r\n "; //HTTP 1.1 Host域不能省略 /*以下头信息域可以省略 $header .= "User-Agent: Mozilla/5.0 (Windows; U; Windows NT 5.1; en-US; rv:1.8.1.13) Gecko/20080311 Firefox/2.0.0.13 \r\n"; $header .= "Accept: text/xml,application/xml,application/xhtml+xml,text/html;q=0.9,text/plain;q=0.8,image/png,q=0.5 \r\n"; $header .= "Accept-Language: en-us,en;q=0.5 "; $header .= "Accept-Encoding: gzip,deflate\r\n"; */ $header .= "Connection:Close\r\n"; if(!empty($cookie)){ $_cookie = strval(NULL); foreach($cookie as $k => $v){ $_cookie .= $k."=".$v."; "; } $cookie_str = "Cookie: " . base64_encode($_cookie) ." \r\n"; //传递Cookie $header .= $cookie_str; } if(!empty($post_data)){ $_post = strval(NULL); foreach($post_data as $k => $v){ $_post .= $k."=".$v."&"; } $post_str = "Content-Type: application/x-www-form-urlencoded\r\n"; $post_str .= "Content-Length: ". strlen($_post) ." \r\n"; //POST数据的长度 $post_str .= $_post."\r\n\r\n "; //传递POST数据 $header .= $post_str; } fwrite($fp, $header); //echo fread($fp, 1024); //服务器返回 fclose($fp); return true; }
このようにして、 fsockopen() 関数を通じて PHP スクリプトの実行をトリガーでき、関数は戻ります。 次に、次のステップに進みます。 ここで問題が発生します。クライアントが切断されると、つまり、triggerRequest がリクエストを送信した後、接続はすぐに閉じられます。これにより、サーバー上で実行されているスクリプトが終了する可能性があり、システムは接続ステータスを維持します。考えられるステータスは 3 つあります: * 0 – NORMAL (正常) * 1 – ABORTED (異常終了) * 2 – TIMEOUT (タイムアウト)
NORMAL 状態で PHP スクリプトが正常に実行される場合、接続は有効です。クライアントが切断されると、ABORTED ステータス フラグがオンになります。リモート クライアント接続の中断は通常、ユーザーが [STOP] ボタンをクリックしたことによって発生します。接続時間が PHP の制限時間 (set_time_limit() 関数を参照) を超えると、TIMEOUT ステータス フラグがオンになります。
クライアントが切断されたときにスクリプトを終了する必要があるかどうかを決定できます。スクリプトの出力を受け入れるリモート ブラウザがない場合でも、スクリプトを完全に実行すると便利な場合があります。デフォルトでは、リモート クライアント接続が中断されるとスクリプトは終了します。この処理は、php.ini のignore_user_abort によって、または Apache .conf 設定の対応する "php_valueignore_user_abort" およびignore_user_abort() 関数によって制御できます。
PHP がユーザーの中断を無視するように指示されていない場合、 register_shutdown_function() で実行がシャットダウンされたとき、つまりスクリプトの実行が完了したとき、または実行が完了したときに呼び出すことができる別の関数を設定できない限り、スクリプトは中断されます。予期せぬ死亡により PHP の実行がシャットダウンされた場合、リモート ユーザーが STOP ボタンをクリックし、スクリプトがデータを再度出力しようとすると、PHP は接続が中断されたことを検出し、シャットダウン トリガー関数を呼び出します。 。
スクリプトは、組み込みのスクリプト タイマーによって中断される場合もあります。デフォルトのタイムアウト制限は 30 秒です。この値は、php.ini で max_execution_time を設定するか、Apache .conf 設定の対応する「php_value max_execution_time」パラメータまたは set_time_limit() 関数を設定することで変更できます。カウンタがタイムアウトすると、上記の接続中断状況と同様にスクリプトが終了し、事前に登録されたシャットダウン トリガー関数もこの時点で実行されます。シャットダウン トリガー関数では、connection_status() 関数を呼び出すことで、タイムアウトによってシャットダウン トリガー関数が呼び出されたかどうかを確認できます。タイムアウトによりシャットダウン トリガー関数が呼び出された場合、関数は 2 を返します。
ABORTED 状態と TIMEOUT 状態が同時に有効になる可能性があることに注意してください。これは、PHP にユーザー終了アクションを無視するように指示する場合に可能です。 PHP はユーザーが切断されたことを認識しますが、スクリプトはまだ実行中です。実行時間制限に達すると、スクリプトは終了し、設定されたシャットダウン トリガー関数も実行されます。この時点で、関数 connection_status() が 3 を返すことがわかります。
したがって、トリガーされるスクリプトにも次のように指定します:
<?php ignore_user_abort(TRUE); //如果客户端断开连接,不会引起脚本abort set_time_limit(0); //取消脚本执行延时上限 或使用: <?php register_shutdown_function(callback fuction[, parameters]); //注册脚本退出时执行的函数