#高い同時実行性に関しては、I/O 多重化をバイパスして特定のプラットフォームを指定する方法はありません。 Linux では、epoll をバイパスする方法はありません。epoll が効率的である理由については説明しません。興味のある学生は、自分で検索して学習してください。PHP で epoll を再生するにはどうすればよいですか?まず、libevent ライブラリをインストールし、次にイベント拡張機能または libevent 拡張機能をインストールする必要があります。そうすれば楽しくプレイできます。
libevent ライブラリと libevent 拡張機能の違いについて混乱している人もいます。簡単に言うと、 libevent ライブラリ これは C 言語での epoll のカプセル化であり、PHP とは何の関係もありません; libevent 拡張機能は、PHP と libevent ライブラリの間の通信ブリッジです。実際、PHP の多くの拡張機能がこれを行います。優れた C 言語ライブラリがいくつかあり、PHP がそれらを直接使用したい場合は、PHP 拡張機能を介して PHP に接続できます。libevent 拡張機能とイベント拡張機能のどちらかを選択します。個人的には、よりオブジェクト指向であるため、イベント拡張機能を好みます。 http://pecl.php.net にアクセスして、お使いの PHP バージョンに対応する拡張機能を検索し、ダウンロードし、コンパイルしてインストールすれば問題ありません。コンピューターにインストールされている複数のバージョンの PHP でコンパイルする場合は、次の点に注意してください。典型的な 5 つの手順は間違いありません:
phpize ./configure make make install php -m | grep event #看看装上了没実装したいサーバーのトランスポート層は TCP プロトコルです。アプリケーション層プロトコルが多すぎます。複雑です。スペースの制限のため、例として HTTP サーバーを簡単に使用します。たとえば、HTTP プロトコル自体は非常に複雑です。実装する詳細が多数あるため、HTTP プロトコルを完全には実装しません。 まず、ソケットを作成します。socket_create、socket_bind、socket_listen の 3 つの手順を実行します。なぜこの 3 つの手順を実行するのでしょうか?それはとても簡単です。トランスポート層プロトコルが何であっても、以下のネットワーク層プロトコルのバージョン (IPV4 または IPV6) を選択する必要があります。トランスポート層の動作方法 (全二重、半二重、またはシンプレックス、TCP) を選択する必要があります。または UDP、どちらかを選択する必要があります。socket_create はこれら 3 つのオプションです。ネットワーク層とトランスポート層を決定した後、socket_bind に対応するどのポートをリッスンするかを指定する必要があります。その後、モニタリングを有効にしてクライアントを指定する必要があります。キューの長さ、これがsocket_listenの機能です。 作成後は、同期ブロッキングは導入しません。プロセスは同時に最大 1 つの接続を保持できます。同時により多くの接続が要求された場合は、待機する必要があります。キューの長さが指定された場合、 .socket_listen を超過したため、504 を返す必要があります。複数のプロセスについても同様で、複数のプロセスが同時に実行されるため、プロセスは高価なリソースであり、プロセスのコンテキスト切り替えには時間と手間がかかり、システム全体の効率が低下します。 それは問題ではありません。私たちは epoll を持っています。何千ものリクエストを保持することも夢ではありません。まず Reactor を実装してください。 libevent ライブラリは Reactor モードであり、関数を直接呼び出すことは Reactor モードを使用するため、PHP で Reactor モードを実装する方法を気にする必要はありません。
<?php use Event; use EventBase; class Reactor { protected $reactor; protected $events; public static $instance = null; const READ = Event::READ | Event::PERSIST; const WRITE = Event::WRITE | Event::PERSIST; public static function getInstance() { if (is_null(self::$instance)) { self::$instance = new self(); self::$instance->reactor = new EventBase; } return self::$instance; } public function add($fd, $what, $cb, $arg = null) { switch ($what) { case self::READ: $event = new Event($this->reactor, $fd, self::READ, $cb, $arg); break; case self::WRITE: $event = new Event($this->reactor, $fd, self::WRITE, $cb, $arg); break; default: $event = new Event($this->reactor, $fd, $what, $cb, $arg); break; } $event->add(); $this->events[(int) $fd][$what] = $event; } public function del($fd, $what = 'all') { $events = $this->events[(int) $fd]; if ($what == 'all') { foreach ($events as $event) { $event->free(); } } else { if ($what != self::READ && $what != self::WRITE) { throw new \Exception('不存在的事件'); } $events[$what]->free(); } } public function run() { $this->reactor->loop(); } public function stop() { foreach ($this->events as $events) { foreach ($events as $event) { $event->free(); } } $this->reactor->stop(); } }上記のコードは非常にシンプルです。概念を簡単に説明します。EventBase は Event インスタンスを含むコンテナーです。このように、上記のコードは非常に理解しやすいです。次に、Server.
<?php use Throwable;use Monolog\Handler\StreamHandler;class Server{ protected $ip; protected $port; protected $socket; protected $reactor; public function __construct($ip, $port) { $this->ip = $ip; $this->port = $port; } public function start() { $socket = $this->createTcpConnection(); stream_set_blocking($socket, false); Reactor::getInstance()->add($socket, Reactor::READ, function($socket) { $conn = stream_socket_accept($socket); stream_set_blocking($conn, false); (new Connection($conn))->handle(); }); Reactor::getInstance()->run(); } public function createTcpConnection() { $schema = sprintf("tcp://%s:%d", $this->ip, $this->port); $socket = stream_socket_server($schema, $errno, $errstr); if ($errno) { throw new \Exception($errstr); } return $socket; } }Connection
<?phpclass Connection{ protected $conn; protected $read_buffer = ''; protected $write_buffer = ''; public function __construct($conn) { $this->conn = $conn; } public function handle() { Reactor::getInstance()->add($this->conn, Reactor::READ, \Closure::fromCallable([$this, 'read'])); } private function read($conn) { $this->read_buffer = ''; if (is_resource($conn)) { while ($content = fread($conn, 65535)) { $this->read_buffer .= $content; } } if ($this->read_buffer) { Reactor::getInstance()->add($conn, Reactor::WRITE, \Closure::fromCallable([$this, 'write'])); } else { Reactor::getInstance()->del($conn); fclose($conn); } } private function write($conn) { if (is_resource($conn)) { fwrite($conn, "HTTP/1.1 200 OK\r\nContent-Type: text/html;charset=utf8\r\nContent-Length:11\r\nConnection: keep-alive\r\n\r\nHello!world"); } } }は、まず 3 つのステップでソケットを作成し、それをノンブロッキング モードに設定します。次に、ソケットを Reactor に追加して、読み取り可能なイベントをリッスンします。読み取り可能とは、バッファーにデータがある場合にのみ読み取ることができることを意味します。読み取り可能なイベントが発生し、新しい接続が来ていることを示します。
stream_socket_accept を使用して新しい接続 Conn を受信し、Conn を Reactor に入れて読み取り可能なイベントを監視します。読み取り可能なイベントが発生した場合は、クライアントがデータが送信されます。データがなくなるまでループで読み取り、Reactor に Conn を入れて書き込み可能なイベントをリッスンします。書き込み可能なイベントが発生した場合、クライアント データが送信されたことを意味します。プロトコルを組み立てて応答を書き込みます。
ab を使用して同時実行性をテストし、
-k パラメーターを追加して接続を多重化します。i5 8G、3W では問題ありません同時実行性 (もちろん、ここにはありません。ディスク I/O、実際の状況では、ディスクからファイルを読み取り、Linux システム コールを介してファイルを読み取ります。また、いくつかのファイル コピー操作があり、比較的コストがかかります)。一般的な解決策は sendfile です。FD から直接別の FD にコピーする必要はありません。効率は比較的高いです。欠点は、PHP にはすぐに実装できる sendfile 拡張機能がないため、自分で行う必要があり、開発コストがかかることです。少し高いです。
これは、PHP で高同時実行サーバーを実装するというアイデアです。EPOLL で解決する限り、アイデアは同じです。これは 3 段階のプロセスです。それを Reactor の下に置いて監視します。 FDイベント。もちろん、これは最も単純なモデルであり、マルチプロセス、コピー nginx、1 つのメインプロセスと N 個のワーカープロセスなど、改善できる点はたくさんあります。マルチプロセスの目的は、マルチコアを使用することです。並行作業。同じことが C 言語の実装にも当てはまりますが、libevent ライブラリを使用して EPOLL を自分でカプセル化することはできません。結局のところ、libevent ライブラリは少し重いため、libevent では多くの機能が使用できません。もちろん C 言語は必要ありません。大量のデータ構造とそのデータ構造上で定義された関数がある オペレーションを記述する必要がある、GC がない、メモリを自分で管理する必要がある、適切な設計が必要である 複数のプロセスを実行する場合、作業が必要IPCのプロセス間通信については、PHPに比べて開発難易度が高く、開発サイクルも非常に長いので、興味があれば学生一人でも遊んでみてはいかがでしょうか。
PHP 関連の技術記事の詳細については、PHP チュートリアル 列にアクセスして学習してください。
以上がPHP を使用した高同時実行サーバーの実装の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

php把负数转为正整数的方法:1、使用abs()函数将负数转为正数,使用intval()函数对正数取整,转为正整数,语法“intval(abs($number))”;2、利用“~”位运算符将负数取反加一,语法“~$number + 1”。

实现方法:1、使用“sleep(延迟秒数)”语句,可延迟执行函数若干秒;2、使用“time_nanosleep(延迟秒数,延迟纳秒数)”语句,可延迟执行函数若干秒和纳秒;3、使用“time_sleep_until(time()+7)”语句。

php除以100保留两位小数的方法:1、利用“/”运算符进行除法运算,语法“数值 / 100”;2、使用“number_format(除法结果, 2)”或“sprintf("%.2f",除法结果)”语句进行四舍五入的处理值,并保留两位小数。

判断方法:1、使用“strtotime("年-月-日")”语句将给定的年月日转换为时间戳格式;2、用“date("z",时间戳)+1”语句计算指定时间戳是一年的第几天。date()返回的天数是从0开始计算的,因此真实天数需要在此基础上加1。

php判断有没有小数点的方法:1、使用“strpos(数字字符串,'.')”语法,如果返回小数点在字符串中第一次出现的位置,则有小数点;2、使用“strrpos(数字字符串,'.')”语句,如果返回小数点在字符串中最后一次出现的位置,则有。

方法:1、用“str_replace(" ","其他字符",$str)”语句,可将nbsp符替换为其他字符;2、用“preg_replace("/(\s|\ \;||\xc2\xa0)/","其他字符",$str)”语句。

php字符串有下标。在PHP中,下标不仅可以应用于数组和对象,还可应用于字符串,利用字符串的下标和中括号“[]”可以访问指定索引位置的字符,并对该字符进行读写,语法“字符串名[下标值]”;字符串的下标值(索引值)只能是整数类型,起始值为0。

查找方法:1、用strpos(),语法“strpos("字符串值","查找子串")+1”;2、用stripos(),语法“strpos("字符串值","查找子串")+1”。因为字符串是从0开始计数的,因此两个函数获取的位置需要进行加1处理。


ホットAIツール

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

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

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

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

人気の記事

ホットツール

Safe Exam Browser
Safe Exam Browser は、オンライン試験を安全に受験するための安全なブラウザ環境です。このソフトウェアは、あらゆるコンピュータを安全なワークステーションに変えます。あらゆるユーティリティへのアクセスを制御し、学生が無許可のリソースを使用するのを防ぎます。

DVWA
Damn Vulnerable Web App (DVWA) は、非常に脆弱な PHP/MySQL Web アプリケーションです。その主な目的は、セキュリティ専門家が法的環境でスキルとツールをテストするのに役立ち、Web 開発者が Web アプリケーションを保護するプロセスをより深く理解できるようにし、教師/生徒が教室環境で Web アプリケーションを教え/学習できるようにすることです。安全。 DVWA の目標は、シンプルでわかりやすいインターフェイスを通じて、さまざまな難易度で最も一般的な Web 脆弱性のいくつかを実践することです。このソフトウェアは、

SublimeText3 英語版
推奨: Win バージョン、コードプロンプトをサポート!

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

SublimeText3 Linux 新バージョン
SublimeText3 Linux 最新バージョン
