(疑似) PHP_PHP のマルチスレッドとマルチプロセッシングのチュートリアル
(擬似)マルチスレッド:外力の助けを借りて
WEBサーバー自体のマルチスレッドを利用して処理を行う WEBサーバーからマルチスレッドプログラムを複数回実装する必要があります。
引用:
PHP 自体はマルチスレッドをサポートしていないことはわかっていますが、WEB サーバーはマルチスレッドをサポートしています。
つまり、複数の人が同時にアクセスできるということは、私が PHP でマルチスレッドを実装するための基礎でもあります。今ファイル a.php を実行しているとしますが、プログラム内で別の b.php を実行するように WEB サーバーにリクエストします。
その後、これら 2 つのファイルが同時に実行されます。(追記: リンクリクエストが送信されると、クライアントが終了したかどうかに関係なく、WEBサーバーはリンクリクエストを実行します)
実行したいのは別のファイルではなく、このファイル内のコードの一部である場合があります。どうすればよいでしょうか?
実際、パラメータを通じて a.php が実行するプログラムを制御できます。
以下の例を見てください:
[php] プレーンコピーを表示
- 関数 runThread(){
- $fp = fsockopen('localhost', 80, $errno, $errmsg );
- fputs($fp, "GET /a.php?act=brnrn");//ここでの 2 番目のパラメーターは、HTTP プロトコルで指定されたリクエスト ヘッダーです。理解できない場合は、RFC の定義を参照してください。 fclose($fp );
- } 関数
- a(){
- $fp = fopen
- ('result_a.log', 'w'); fputs($fp, ' で設定します
- . Date('h:i:s', time( )) . )マイクロタイム() ."rn"); fclose($fp); } 関数 b(){
- $fp =
- fopen (
- 'result_b.log', 'w' );
- fputs($fp, ' で設定します . Date('h:i:s'
- , time( )) . )マイクロタイム() ."rn"); fclose($fp); } if(!isset($_GET['act'])){
- $_GET['act'] =
- ' ;};
- if($_GET['act'] == 'a'){ runThread();
- ; } else
- if($_GET['act'] == 'b'){
- b();
- }; ?>
-
result_a.log と result_b.log を開いて、2 つのファイルのアクセス時間を比較すると、これら 2 つが実際に異なるスレッドで実行されていることがわかります。
まったく同じ時間もあります。
上記は単なる例であり、他の形式に改良することができます。
PHP でマルチスレッドが使用できるようになったので、問題が発生します。それは、PHP 自体がマルチスレッドをサポートしていないことです。したがって、 のようなものはありません。
Java の同期メソッドでは、どうすればよいでしょうか。
1. 競合を避けるために同じリソースにアクセスしないようにしてください。ただし、データベースは同時操作をサポートしているため、マルチスレッド PHP では同時に操作することができます。
同じファイルにデータを書き込まないでください。書き込む必要がある場合は、flock を呼び出してファイルをロックするなど、他の方法を使用するか、一時ファイルを作成してください。そして別のスレッドでこのファイルが消えるのを待ちます while(file_exits('xxx')); これは、この一時ファイルが存在するということは、スレッドが実際に動作していることを意味します
このファイルがもう存在しない場合は、他のスレッドがそのファイルを解放したことを意味します。
2. fputs の実行後に runThread が取得するソケットからデータを読み取らないようにします。これは、fgets と同様に、ノンブロッキング モードを使用する必要があるためです。そのような関数はすぐに戻ります。そのため、ブロックモードが使用されている場合、プログラムは実行する前に上記の戻りを待つ必要があります
。データを交換する必要がある場合は、最終的に外部ファイルまたはデータを使用してそれを実現します。
ここまで述べましたが、これには実際的な意味があるのでしょうか?この方法を使用する必要があるのはどのような場合ですか?
答えは「はい」です。ご存知のとおり、ネットワーク リソースを常に読み取るアプリケーションでは、ネットワークの速度がボトルネックになります。この形式を採用すると、複数のスレッドを同時に使用できます。別のページを読んでください。
8848やsoasoなどのショッピングモールサイトから情報を検索できるプログラムを作りました。アリババのウェブサイトからビジネス情報や企業ディレクトリを読み取るプログラムもあります
マルチプロセス: PHPのプロセス制御関数(PCNTL/スレッド制御関数)を使用します
関数のリファレンスは http://www.php.net/manual/zh/ref.pcntl.phpにあります。 Unix Like OS でのみ使用でき、Windows では使用できません。
php をコンパイルするときは、--enable-pcntl を追加する必要があります。これは、WEB サーバー環境ではなく、CLI モードでのみ実行することをお勧めします。以下は短いテストコードです:
[php] プレーンコピーを表示- 宣言(ティック=1); $bWaitFlag
- = FALSE; /// プロセスの終了を待つかどうか $intNum
- = 10; /// プロセスの総数 $pids
- = array(); エコー (「スタート」);
- for($i = 0; $i <
- $intNum; $i++) { $pids[$i] = pcntl_fork();/// 子プロセスを生成し、現在の行からテスト実行コードを開始します。子プロセスのデータ情報は継承しません。親プロセス
- if(!$pids[$i]) {
- //サブプロセスプロセスコードsegment_Start $str=
- ""; 睡眠(5+
- $i); ($j=0;$j
- <$i;$j
- ++) { $str .= "*" ;} echo "$i -> " . time() . " $str n"; 終了();
- //サブプロセスプロセスコードsegment_End } }
- if ($bWaitFlag)
- {
- for(
- $i = 0;
- $i pcntl_waitpid($pids[
- $i],
- $status, WUNTRACED ); と } } エコー (「終わります」
- );
実行結果は以下の通りです:
コード:[クリップボードにコピー][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
時間的な関係でのより詳細な研究はまだ行われていないため、私が提示した手の接続を参照してください。 [文章二] 尝试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] プレーンコピーを表示- if($argc==1){
- echo("argvn");
- }
- $arg = $argv[1];
- for($i=0; $i<10; $i++)
- {
- echo($i.".1.".time()." exec $arg n");
- if($arg=='php2'){
- 睡眠(1);
- echo($i.".2.".time()." exec $arg n");
- 睡眠(1);
- }else{
- 睡眠(1);
- }
- }
- ?>
主调用子程序、他调用子程程、同時に発行された収集子程序の出力
[php] プレーンコピーを表示- エラー報告(E_ALL);
- $handle1 = popen('php sub.php php1', 'r');
- $handle2 = popen('php sub.php php2', 'r');
- $handle3 = popen('php sub.php php3', 'r');
- echo "'$handle1'; " 。 gettype($handle1) 。 「ん」;
- echo "'$handle2'; " 。 gettype($handle2) 。 「ん」;
- echo "'$handle3'; " 。 gettype($handle3) 。 「ん」;
- //睡眠(20);
- ながら(!feof($handle1) || !feof($handle2) || ! feof($ハンドル3 ) )
- {
- $read = fgets($handle1);
- エコー $read;
- $read = fgets($handle2);
- エコー $read;
- $read = fgets($handle3);
- エコー $read;
- }
- pclose($handle1);
- pclose($handle2);
- pclose($handle3);
次は我机上の出力:
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] プレーンコピーを表示
- /*
- メインタスクマネージャー
- 同時実行サブタスクリスト
- */
- include("../common/conf.php"); include
- ("../common/function.php"); //開いているプロセスの数
- $exec_number = 40 ;
- /***** 主要 ********/
- if($argc
- ==1){ エコー("argvn"
- ); }
- $タスクファイル = $argv[1]; //タスクリスト
- $タスクリスト
- = ファイル($タスクファイル); $tasklist_len =
- count($tasklist); $tasklist_pos = 0; $handle_list
- = array(); ながら
- (1) { //子プロセスリストが空いている場合は、子プロセスリストに記入します
- for($i =
- $tasklist_pos$i<$tasklist_len; ;) M om $コマンド =
- $タスクリスト [ $ i]; $ handle_list [] = Popen ( $ commit
- , "r"
- ); tolog("begin task t ".$tasklist[$i]); ️
- ~ }
- //子プロセスリストが空の場合は終了します if(0 == count
- ($handle_list)) ️ 。 }
- //子プロセスリストの出力を確認し、停止した子プロセスを閉じて記録します
- $end_handle_keys = array();
- // $ str 」 ️ pclose($handle); } //停止した子プロセスをキックアウトします
- foreach( $end_handle_keys
- as $key) (設定解除 (n $ hand_list [ $ key
- ]); //var_dump($handle_list); //exit;
- } } tolog(「ん*************************終わり****************** ******nn "
- 、""
- 、本当);
Socket マルチプロセス受信用のコードを添付します: [php] プレーンコピーを表示
- する {
- if (($msgsock = socket_accept($sock)) < 0) {
- echo "socket_accept() 失敗しました: 理由: " 。 socket_strerror($msgsock) 。 「ん」;
- 休憩;
- }
- $pid = pcntl_fork();
- if ($pid == -1) {
- 死ぬ(「フォークできませんでした」);
- } else if (!$pid) {
- ....
- socket_write($msgsock, $msg, strlen($msg));
- do {
- ......
- } ながら (本当);
- socket_close($msgsock);
- }
- } ながら (本当);
翻訳地址:http://www.alixixi.com/program/a/2008050731686.shtml

ホットAIツール

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

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

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

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

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

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

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

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

ホットトピック











PHP 8.4 では、いくつかの新機能、セキュリティの改善、パフォーマンスの改善が行われ、かなりの量の機能の非推奨と削除が行われています。 このガイドでは、Ubuntu、Debian、またはその派生版に PHP 8.4 をインストールする方法、または PHP 8.4 にアップグレードする方法について説明します。

ファイルのアップロードを行うには、フォーム ヘルパーを使用します。ここではファイルアップロードの例を示します。

CakePHP は、PHP 用のオープンソース フレームワークです。これは、アプリケーションの開発、展開、保守をより簡単にすることを目的としています。 CakePHP は、強力かつ理解しやすい MVC のようなアーキテクチャに基づいています。モデル、ビュー、コントローラー

Visual Studio Code (VS Code とも呼ばれる) は、すべての主要なオペレーティング システムで利用できる無料のソース コード エディター (統合開発環境 (IDE)) です。 多くのプログラミング言語の拡張機能の大規模なコレクションを備えた VS Code は、
