ホームページ >バックエンド開発 >PHPチュートリアル >(疑似) PHP_PHP のマルチスレッドとマルチプロセッシングのチュートリアル
(擬似)マルチスレッド:外力の助けを借りて
WEBサーバー自体のマルチスレッドを利用して処理を行う WEBサーバーからマルチスレッドプログラムを複数回実装する必要があります。
引用:
PHP 自体はマルチスレッドをサポートしていないことはわかっていますが、WEB サーバーはマルチスレッドをサポートしています。
つまり、複数の人が同時にアクセスできるということは、私が PHP でマルチスレッドを実装するための基礎でもあります。今ファイル a.php を実行しているとしますが、プログラム内で別の b.php を実行するように WEB サーバーにリクエストします。
その後、これら 2 つのファイルが同時に実行されます。(追記: リンクリクエストが送信されると、クライアントが終了したかどうかに関係なく、WEBサーバーはリンクリクエストを実行します)
実行したいのは別のファイルではなく、このファイル内のコードの一部である場合があります。どうすればよいでしょうか?
実際、パラメータを通じて a.php が実行するプログラムを制御できます。
以下の例を見てください:
[php] プレーンコピーを表示
Java の同期メソッドでは、どうすればよいでしょうか。
1. 競合を避けるために同じリソースにアクセスしないようにしてください。ただし、データベースは同時操作をサポートしているため、マルチスレッド PHP では同時に操作することができます。
同じファイルにデータを書き込まないでください。書き込む必要がある場合は、flock を呼び出してファイルをロックするなど、他の方法を使用するか、一時ファイルを作成してください。そして別のスレッドでこのファイルが消えるのを待ちます while(file_exits('xxx')); これは、この一時ファイルが存在するということは、スレッドが実際に動作していることを意味します
このファイルがもう存在しない場合は、他のスレッドがそのファイルを解放したことを意味します。
2. fputs の実行後に runThread が取得するソケットからデータを読み取らないようにします。これは、fgets と同様に、ノンブロッキング モードを使用する必要があるためです。そのような関数はすぐに戻ります。そのため、ブロックモードが使用されている場合、プログラムは実行する前に上記の戻りを待つ必要があります
。データを交換する必要がある場合は、最終的に外部ファイルまたはデータを使用してそれを実現します。
ここまで述べましたが、これには実際的な意味があるのでしょうか?この方法を使用する必要があるのはどのような場合ですか?
答えは「はい」です。ご存知のとおり、ネットワーク リソースを常に読み取るアプリケーションでは、ネットワークの速度がボトルネックになります。この形式を採用すると、複数のスレッドを同時に使用できます。別のページを読んでください。
このテクノロジー。 どちらのプログラムも、情報を読み取ってデータベースに保存するためにサーバーに継続的に接続する必要があるためです。 このテクニックを利用すると、応答を待っている間にボトルを削除するだけです
首。
マルチプロセス: PHPのプロセス制御関数(PCNTL/スレッド制御関数)を使用します
関数のリファレンスは http://www.php.net/manual/zh/ref.pcntl.phpにあります。 Unix Like OS でのみ使用でき、Windows では使用できません。
php をコンパイルするときは、--enable-pcntl を追加する必要があります。これは、WEB サーバー環境ではなく、CLI モードでのみ実行することをお勧めします。
以下は短いテストコードです:
[php] プレーンコピーを表示
コード:[クリップボードにコピー][qiao@oicq qiao]$ phptest.php
スタート
終了
[qiao@oicq qiao]$ ps -aux | grep "php"
喬 32275 0.0 0.5 49668 6148pts/1 S 14:03 0:00/usr/local/php4/b
喬 32276 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
喬 32277 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
喬 32278 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
喬 32279 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
喬 32280 0.0 0.5 49668 6152pts/1 S 14:03 0:00 /usr/local/php4/b
喬 32281 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
喬 32282 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
喬 32283 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
喬 32284 0.0 0.5 49668 6152pts/1 S 14:03 0:00/usr/local/php4/b
qiao 32286 0.0 0.0 1620 600pts/1 S 14:03 0:00 grep php
[qiao@oicq qiao]$ 0 -> 1133503401
1 -> 1133503402 *
2 -> 1133503403 **
3 -> 1133503404 ***
4 -> 1133503405 ****
5 -> 1133503406 *****
6 -> 1133503407 ******
7 -> 1133503408 *******
8 -> 1133503409 ********
9 -> 1133503410 *********
[qiao@oicq qiao]$
結果$bWaitFlag=TURE、結果は以下の通り:
コード:[クリップボードにコピー][qiao@oicq qiao]$ phptest.php
スタート
0 -> 1133503602
0 待つ -> 1133503602
1 -> 1133503603 *
待つ 1 -> 1133503603
2 -> 1133503604 **
2 待ってください -> 1133503604
3 -> 1133503605 ***
3 待つ -> 1133503605
4 -> 1133503606 ****
4 待つ -> 1133503606
5 -> 1133503607 *****
5 待つ -> 1133503607
6 -> 1133503608 ******
6 待つ -> 1133503608
7 -> 1133503609 *******
7 待つ -> 1133503609
8 -> 1133503610 ********
8 待つ -> 1133503610
9 -> 1133503611 *********
9 待つ -> 1133503611
終了
[qiao@oicq qiao]$
从
多くのプロセスの例を見ると、pcntl_fork() の使用後に子プロセスが生成され、その子プロセスが実行される代コードが pcntl_fork() の後の代コードから始まります。
プロセスは父プロセスのデータ情報を継承しないため (実際には、父プロセスのデータは完全に新しい記録を作成します)、そのため、 if(!$pids[$i]) を使用して子プロセスの実行のコード セグメントを制御します。
作者:デュラオ5
除了フォーク、cli下の公開方式にも一種類、見られる例:
以下は例です:
被管理用子プログラム代コード:
[php] プレーンコピーを表示
主调用子程序、他调用子程程、同時に発行された収集子程序の出力
[php] プレーンコピーを表示
次は我机上の出力:
C:my_hunter>php exec.php
'リソースID #4';リソース
'リソースID #5';リソース
'リソースID #6';リソース
0.1.1147935331 php1 を実行
0.1.1147935331 php2 を実行
0.1.1147935331 php3 を実行
1.1.1147935332 php1 を実行
0.2.1147935332 php2 を実行
1.1.1147935332 php3 を実行
2.1.1147935333 php1 を実行
1.1.1147935333 php2 を実行
2.1.1147935333 php3 を実行
3.1.1147935334 php1 を実行
1.2.1147935334 php2 を実行します
3.1.1147935334 php3 を実行
4.1.1147935335 php1 を実行
2.1.1147935335 php2 を実行
4.1.1147935335 php3 を実行
5.1.1147935336 php1 を実行
2.2.1147935336 php2 を実行
5.1.1147935336 php3 を実行
6.1.1147935337 php1 を実行
3.1.1147935337 php2 を実行
6.1.1147935337 php3 を実行
7.1.1147935338 php1 を実行
3.2.1147935338 php2 を実行
7.1.1147935338 php3 を実行
8.1.1147935339 php1 を実行します
4.1.1147935339 php2 を実行します
8.1.1147935339 php3 を実行
9.1.1147935340 php1 を実行
4.2.1147935340 php2 を実行
9.1.1147935340 php3 を実行
5.1.1147935341 php2 を実行します
5.2.1147935342 php2 を実行
6.1.1147935343 php2 を実行します
6.2.1147935344 php2 を実行します
7.1.1147935345 php2 を実行します
7.2.1147935346 php2 を実行
8.1.1147935347 php2 を実行します
8.2.1147935348 php2 を実行します
9.1.1147935349 php2 を実行します
9.2.1147935350 php2 を実行
**总结:**
** メインプログラム循環等の子プロセスを待機し、fgets または fread によって子プロセスの出力が取得され、時間的に確認されて実行されます。**
------------------------------------------------------------
その後の変更点:
* Popen を開くハンドルは一方向であり、子プロセスのやり取りが必要な場合は、proc_open
を使用できます。
* 数組和子関数数を使用する代わりに(!feof($handle1)|| !feof($handle2) || !feof($handle3) ) この写法
* fread を使用すると、1 行ごとではなく、子プロセスによって生成された出力が完了します。
[php] プレーンコピーを表示
翻訳地址:http://www.alixixi.com/program/a/2008050731686.shtml