PHP は Nginx、Apache などの特定の WEB サーバーで実行する必要があることは誰もが知っていますが、PHP はどのように起動し、サーバー内でどのように実行され、両者はどのように相互作用するのでしょうか?
1. WEB サーバーは PHP インターフェイスを呼び出します
Apache サーバーを例として、サーバーがどのように PHP を起動し、PHP のメソッドを呼び出すかを見てみましょう。 Apache サーバーが起動して PHP を実行すると、通常は mod_php7
モジュール (php5.* バージョンの場合は mod_php5
モジュールとモジュールのサフィックス名) を通じて統合されます。 mod_php7 の構造は次のとおりです (ソース コードのパスは php/sapi/apache2handler/mod_php7.c です):
AP_MODULE_DECLARE_DATA module php7_module = { STANDARD20_MODULE_STUFF,/* 宏,包括版本,版本,模块索引,模块名,下个模块指针等信息 */ create_php_config, /* create per-directory config structure */ merge_php_config, /* merge per-directory config structures */ NULL, /* create per-server config structure */ NULL, /* merge per-server config structures */ php_dir_cmds, /* 模块定义的所有指令 */ php_ap2_register_hook /* register hooks */ };
Apache が PHP でメソッドを呼び出す必要がある場合のみ、 mod_php7 モジュールを通じてリクエストを PHP に伝える必要があります。PHP 層 処理後、データは Apache に返され、プロセス全体が終了します (追加してください: Apache サーバーが PHP を開始するとき、実際には 2 つのロード方法があります。先ほど説明した mod_php5 モジュールのロード方法は、静的ロードとして理解できます。つまり、PHP をロードするには Apache サーバーを再起動する必要がありますが、動的ロードではサーバーを再起動する必要はありません。 PHP 起動の目的を達成するには、シグナルを送信して PHP 固定モジュールをサーバーにロードするだけで済みます。ただし、動的ロードの前に、ロード モジュールをダイナミック リンク ライブラリにコンパイルし、その後サーバーの構成ファイルに設定する必要があります)。 PHP における Apache のモデル構造は上に示されていますが、Apache サーバー内の対応するモジュール構造は次のとおりです (ソース コードは Apache にあります。以下同様):
struct module_struct { int version; int minor_version; int module_index; const char *name; void *dynamic_load_handle; struct module_struct *next; unsigned long magic; void (*rewrite_args) (process_rec *process); void *(*create_dir_config) (apr_pool_t *p, char *dir); void *(*merge_dir_config) (apr_pool_t *p, void *base_conf, void *new_conf); void *(*create_server_config) (apr_pool_t *p, server_rec *s); void *(*merge_server_config) (apr_pool_t *p, void *base_conf, void *new_conf); const command_rec *cmds; void (*register_hooks) (apr_pool_t *p); }
php7_module を参照してください。と module_struct まだ大きな違いがありますが、マクロ php7_module.STANDARD20_MODULE_STUFF が定義されている方法を見ると、この 2 つの構造は非常に似ていると思うかもしれません。実際、このマクロは module_struct の最初の 8 つのパラメータを定義しており、次のように定義されています。
#define STANDARD20_MODULE_STUFF MODULE_MAGIC_NUMBER_MAJOR, \ MODULE_MAGIC_NUMBER_MINOR, \ -1, \ __FILE__, \ NULL, \ NULL, \ MODULE_MAGIC_COOKIE, \ NULL /* rewrite args spot */
次に、php7_module.php_dir_cmds でモジュールのすべての命令セットを定義します。具体的な定義内容は次のとおりです (コードパスは php/sapi/apache2handler/apache_config.c):
const command_rec php_dir_cmds[] = { AP_INIT_TAKE2("php_value", php_apache_value_handler, NULL, OR_OPTIONS, "PHP Value Modifier"), AP_INIT_TAKE2("php_flag", php_apache_flag_handler, NULL, OR_OPTIONS, "PHP Flag Modifier"), AP_INIT_TAKE2("php_admin_value", php_apache_admin_value_handler, NULL, ACCESS_CONF|RSRC_CONF, "PHP Value Modifier (Admin) "), AP_INIT_TAKE2("php_admin_flag", php_apache_admin_flag_handler, NULL, ACCESS_CONF|RSRC_CONF, "PHP Flag Modifier (Admin)"), AP_INIT_TAKE1("PHPINIDir", php_apache_phpini_set, NULL, RSRC_CONF, "Directory containing the php.ini file"), {NULL} };
つまり、PHP層のみ 上記5つの命令がApacheに提供されます。 各命令の実装ソースコードもapache_config.cファイルにあります。 最後にphp7_module.php_ap2_register_hookだけが残ります。 その定義は次のとおりです(コードパスは php/sapi/apache2handler/mod_php7.c) :
void php_ap2_register_hook(apr_pool_t *p) { ap_hook_pre_config(php_pre_config, NULL, NULL, APR_HOOK_MIDDLE); ap_hook_post_config(php_apache_server_startup, NULL, NULL, APR_HOOK_MIDDLE); ap_hook_handler(php_handler, NULL, NULL, APR_HOOK_MIDDLE); #ifdef ZEND_SIGNALS ap_hook_child_init(zend_signal_init, NULL, NULL, APR_HOOK_MIDDLE); #endif ap_hook_child_init(php_apache_child_init, NULL, NULL, APR_HOOK_MIDDLE); }
php7_module.php_ap2_register_hook 関数には 4 つのフックと対応する処理関数が含まれています。pre_config、pre_config、post_config、child_init は起動フックであり、ハンドラー フックはリクエスト フックであり、サーバー リクエストは呼び出しであり、これらのフックを通じて、Apache サーバー経由で PHP を開始できます。
この時点で、WEB サーバーがどのように PHP を起動し、PHP のメソッドを呼び出すのかをすでに理解しているはずです。次に、PHP がどのように WEB サーバー インターフェイスを呼び出すかを説明します。
2.PHP は WEB サーバー インターフェイスを呼び出します
この問題について説明する前に、SAPI とは何かを理解する必要があります。 SAPI は、実際にはサーバー抽象化層とサーバー抽象化層の間で守られる共通の取り決めであり、PHP がキャッシュのクリアなど、サーバー内のメソッドを呼び出す必要がある場合に、キャッシュをクリアする実装メソッドが実装されるため、容易に理解できます。サーバー内でこのメソッドを呼び出すと、PHP 層はそれをまったく認識しません。サーバー内でこのメソッドを呼び出すにはどうすればよいですか?どうすればよいですか?このとき、両当事者が合意を形成する必要があり、その後、サーバーは合意されたインターフェイスのセットを PHP に提供します。サーバー抽象化レイヤーとのこれらの共通の合意を SAPI インターフェイスと呼びます。
問題は、サーバー Apache に対しては SAPI のセットを提供できますが、次回別のサーバーまたは他の「サードパーティ」が登場した場合、それらにも SAPI のセットを提供する必要があるかということです。 ? 別個の SAPI についてはどうですか?私たちの賢い PHP 開発者は、これ、つまり、すべての「サードパーティ」に共通の SAPI インターフェイスのセットを提供することを考えたに違いありません。しかし、新しい「サードパーティ」がそのインターフェイスを必要とする場合、共通の SAPI は必要ないのではないかと疑問に思うかもしれません。 , どうすればよいでしょうか? 私の理解では、PHP の一般的な SAPI インターフェイスに新しい機能を追加することです。これは単なる個人的な意見です。一般的な SAPI 構造は次のとおりです (ソース コード パス: php/main/SAPI.h):
struct _sapi_module_struct { char *name; // 名字 char *pretty_name; // 更好理解的名字 int (*startup)(struct _sapi_module_struct *sapi_module); // 启动函数 int (*shutdown)(struct _sapi_module_struct *sapi_module); // 关闭函数 int (*activate)(TSRMLS_D); // 激活 int (*deactivate)(TSRMLS_D); // 停用 void (*flush)(void *server_context); // flush char *(*read_cookies)(TSRMLS_D); //read Cookies //... };
この構造には多くの変数があるため、1 つずつリストすることはしません。内部の変数について簡単に説明します。スタートアップ関数は SAPI の初期化時に呼び出され、シャットダウン関数はデータ構造を解放するために使用されます。 SAPI のメモリ、read_cookie SAPI 起動時に呼び出され、この関数で取得した値を SG(request_info).cookie_data に代入します。では、PHP によって提供される一般的な SAPI の場合、Apache サーバーは独自のインターフェイスをどのようにカスタマイズするのでしょうか?具体的な構造は次のとおりです (ソース コードのパスは php/sapi/apache2handler/sapi_apache2.c です):
static sapi_module_struct apache2_sapi_module = { "apache2handler", "Apache 2.0 Handler", php_apache2_startup, /* startup */ php_module_shutdown_wrapper, /* shutdown */ NULL, /* activate */ NULL, /* deactivate */ php_apache_sapi_ub_write, /* unbuffered write */ php_apache_sapi_flush, /* flush */ php_apache_sapi_get_stat, /* get uid */ php_apache_sapi_getenv, /* getenv */ php_error, /* error handler */ php_apache_sapi_header_handler, /* header handler */ php_apache_sapi_send_headers, /* send headers handler */ NULL, /* send header handler */ php_apache_sapi_read_post, /* read POST data */ php_apache_sapi_read_cookies, /* read Cookies */ php_apache_sapi_register_variables, php_apache_sapi_log_message, /* Log message */ php_apache_sapi_get_request_time, /* Request Time */ NULL, /* Child Terminate */ STANDARD_SAPI_MODULE_PROPERTIES };
上記のソース コード ディレクトリ php/sapi/apache2handler/ 内で、ディレクトリ php/ の下にあるものはすべてsapi は SAPI を通じて呼び出されます。「サードパーティ」のディレクトリ構造は、次の図に示すとおりです。ディレクトリ php/sapi/apache2handler には、PHP と対話するためのインターフェイスが含まれています。sapi_apache2.c は、PHP と Apache の間で合意された SAPI インターフェイス ファイルです。 。 ##################################
看到这里,大家应该基本清楚PHP层是怎样调用服务器层的接口,为了巩固上面的知识,下面举个栗子,即在Apache服务器环境下读取cookie:
SG(request_info).cookie_data = sapi_module.read_cookies(TSRMLS_C);
对于任意一个服务器在加载时,我们都会指定sapi_module,Apache的sapi_module是apache2_sapi_module,它的read_cookies方法的是php_apache_sapi_read_cookies函数,这样就实现PHP层调用Apache的接口,是不是很简单呢:)
更多相关问题可以访问PHP中文网相关教程:https://www.php.cn/course/list/29/type/2.html
以上がPHP の原則と PHP と WEB サーバー間の相互作用についての深い理解の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

この記事では、酸とベースのデータベースモデルを比較し、その特性と適切なユースケースを詳述しています。酸は、財務およびeコマースアプリケーションに適したデータの整合性と一貫性を優先し、ベースは可用性に焦点を当て、

この記事では、コードインジェクションのような脆弱性を防ぐために、PHPファイルのアップロードを確保することについて説明します。ファイルタイプの検証、セキュアストレージ、およびアプリケーションセキュリティを強化するエラー処理に焦点を当てています。

記事では、組み込み関数、ホワイトリストアプローチ、サーバー側の検証などの手法に焦点を当てたセキュリティを強化するためのPHP入力検証のベストプラクティスについて説明します。

この記事では、Token BucketやLeaky BucketなどのアルゴリズムやSymfony/Rate-Limiterなどのライブラリを使用するなど、PHPでAPIレート制限を実装するための戦略について説明します。また、監視、動的に調整されたレートの制限、および手をカバーします

この記事では、パスワードを保護するためにPHPでpassword_hashとpassword_verifyを使用することの利点について説明します。主な議論は、これらの関数が自動塩の生成、強力なハッシュアルゴリズム、およびSecurを通じてパスワード保護を強化するということです

この記事では、PHPおよび緩和戦略におけるOWASPトップ10の脆弱性について説明します。重要な問題には、PHPアプリケーションを監視および保護するための推奨ツールを備えたインジェクション、認証の壊れ、XSSが含まれます。

この記事では、PHPでのXSS攻撃を防ぐための戦略について説明し、入力の消毒、出力エンコード、セキュリティを向上させるライブラリとフレームワークの使用に焦点を当てています。

この記事では、PHPでのインターフェイスと抽象クラスの使用について説明し、それぞれをいつ使用するかに焦点を当てています。インターフェイスは、無関係なクラスや複数の継承に適した、実装なしで契約を定義します。抽象クラスは共通の機能を提供します


ホットAIツール

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

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

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

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

人気の記事

ホットツール

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

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

MantisBT
Mantis は、製品の欠陥追跡を支援するために設計された、導入が簡単な Web ベースの欠陥追跡ツールです。 PHP、MySQL、Web サーバーが必要です。デモおよびホスティング サービスをチェックしてください。

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

mPDF
mPDF は、UTF-8 でエンコードされた HTML から PDF ファイルを生成できる PHP ライブラリです。オリジナルの作者である Ian Back は、Web サイトから「オンザフライ」で PDF ファイルを出力し、さまざまな言語を処理するために mPDF を作成しました。 HTML2FPDF などのオリジナルのスクリプトよりも遅く、Unicode フォントを使用すると生成されるファイルが大きくなりますが、CSS スタイルなどをサポートし、多くの機能強化が施されています。 RTL (アラビア語とヘブライ語) や CJK (中国語、日本語、韓国語) を含むほぼすべての言語をサポートします。ネストされたブロックレベル要素 (P、DIV など) をサポートします。
