Maison >développement back-end >tutoriel php >Explication détaillée du mécanisme de fonctionnement sous-jacent de PHP
Dans cet article, nous partageons principalement avec vous une explication détaillée du mécanisme de fonctionnement sous-jacent de PHP. Tout d'abord, nous partagerons avec vous les concepts de conception et les caractéristiques de PHP, le système à quatre couches de PHP, etc. cela peut vous aider.
Modèle multi-processus : PHP étant un modèle multi-processus, les différentes requêtes n'interfèrent pas les unes avec les autres, ce qui garantit cette requête sera suspendue. La perte n'affectera pas le service global. Bien entendu, avec l'évolution des temps, PHP a déjà pris en charge le modèle multi-thread.
Langage faiblement typé : contrairement à C/C++, Java, C# et d'autres langages, PHP est un langage faiblement typé. Le type d'une variable n'est pas déterminé au début. Il est déterminé lors du fonctionnement et une conversion de type implicite ou explicite peut se produire. La flexibilité de ce mécanisme est très pratique et efficace dans le développement Web. Les détails seront discutés plus tard dans les variables PHP. sont détaillés dans.
Langage interprété : PHP est différent de C/C++, Java, C# et d'autres langages compilés dans les étapes en cours. PHP doit être analysé dans un langage compilé via lexical et syntaxique. analyse d'abord. , pour courir! Par conséquent, PHP n'est pas adapté aux applications à grande échelle telles que les calculs de hautes performances ou de Big Data. Bien qu'il n'y ait aucune différence entre 0,001 seconde et 0,1 seconde pour les utilisateurs de navigateur, il ne convient pas aux autres domaines
Comme vous pouvez le voir sur l'image, PHP implémente un processus d'exécution dynamique typique d'un langage : après avoir obtenu un morceau de code, après avoir traversé des étapes telles que l'analyse lexicale et l'analyse syntaxique, le programme source sera traduit en instructions (opcodes) . La machine virtuelle ZEND exécute ensuite ces instructions en séquence pour terminer l'opération. PHP lui-même est implémenté en C, donc les fonctions finalement appelées sont toutes des fonctions C. En fait, on peut considérer PHP comme un logiciel développé en C.
Le cœur de l'exécution de PHP réside dans les instructions traduites, qui sont des opcodes.
L'Opcode est l'unité la plus basique de l'exécution d'un programme PHP. Un opcode se compose de deux paramètres (op1, op2), d'une valeur de retour et d'une fonction de traitement. Le programme PHP se traduit finalement par l'exécution séquentielle d'un ensemble de fonctions de traitement d'opcodes.
Plusieurs fonctions de traitement courantes :
PHP
|
ZEND_ASSIGN_SPEC_CV_CV_HANDLER : allocation de variable ($a=$b)ZEND_DO_FCALL_BY_NAME_SPEC_HANDLER : appel de fonctionZEND_CONCAT_SPEC_CV_CV_HANDLER : concaténation de chaîne $a.$bZEND_ADD_SPEC_ CV_CONST_HANDLER : Opération d'ajout $a+2ZEND_IS_EQUAL_SPEC_CV_CONST : Juge égal $a==1ZEND_IS_IDENTICAL_SPEC_CV_CONST : Juge égal $a===1 | tr>
HashTable est la structure de données de base de zend. Elle est utilisée pour implémenter presque toutes les fonctions courantes de PHP. Le tableau PHP que nous connaissons est son application typique. De plus, dans Zend, la table des symboles de fonction, les variables globales, etc. sont également implémentées sur la base de la table de hachage.
La table de hachage de PHP présente les fonctionnalités suivantes :
Prend en charge les requêtes clé>valeur typiques
peut être utilisée comme un tableau Utiliser
pour ajouter et supprimer des nœuds est une complexité O(1)
la clé prend en charge les types mixtes : il existe des tableaux associatifs et des index combinés tableaux en même temps
La valeur prend en charge les types mixtes : tableau ("string",2332)
prend en charge le parcours linéaire : comme foreach
La table de hachage Zend implémente la structure de hachage typique de la table de hachage, et fournit en même temps la fonction de parcours avant et arrière du tableau en attachant une liste doublement chaînée. Sa structure est la suivante :
Comme vous pouvez le voir, il existe à la fois des structures de hachage sous forme de clé->valeur et un mode liste doublement chaînée dans la table de hachage, ce qui le rend très pratique pour la recherche rapide et le parcours linéaire.
Structure de hachage : la structure de hachage de Zend est un modèle de table de hachage typique, qui utilise une liste chaînée pour résoudre les conflits. Il convient de noter que la table de hachage de Zend est une structure de données à croissance automatique Lorsque le nombre de tables de hachage est plein, elle s'étendra dynamiquement de 2 fois et repositionnera les éléments. La taille initiale est de 8. De plus, lors de l'exécution d'une recherche rapide clé->valeur, zend lui-même a également apporté quelques optimisations pour accélérer le processus en échangeant de l'espace contre du temps. Par exemple, une variable nKeyLength est utilisée dans chaque élément pour identifier la longueur de la clé pour une détermination rapide.
Liste chaînée double : la table de hachage Zend implémente un parcours linéaire d'éléments via une structure de liste chaînée. En théorie, il suffit d'utiliser une liste chaînée unidirectionnelle pour le parcours. L'objectif principal de l'utilisation d'une liste chaînée bidirectionnelle est de supprimer rapidement et d'éviter le parcours. La table de hachage Zend est une structure composite lorsqu'elle est utilisée comme tableau, elle prend en charge les tableaux associatifs communs et peut également être utilisée comme numéros d'index séquentiels, et permet même un mélange des deux.
Tableau associatif PHP : le tableau associatif est une application hash_table typique. Un processus de requête passe par les étapes suivantes (comme le montre le code, il s'agit d'un processus de requête de hachage courant et quelques jugements rapides sont ajoutés pour accélérer la recherche.) :
PHP
|
getKeyHashValueh;index=n&nTableMask;Bucket*p=arBucket[index];while( p){if((p->h==h)&(p->nKeyLength==nKeyLength)){RETURNp-> données; }p=p->next;}RETOURFAUTE; |
Tableau d'index PHP : Le tableau d'index est notre tableau commun, accessible via des indices. Par exemple, $arr[0], Zend HashTable est normalisé en interne, et la valeur de hachage et nKeyLength (0) sont également affectées à la clé de type d'index. La variable membre interne nNextFreeElement est l'identifiant maximum actuellement attribué, qui est automatiquement augmenté de un après chaque poussée. C'est ce processus de normalisation qui permet à PHP de réaliser un mélange de données associatives et non associatives. En raison de la particularité de l'opération push, l'ordre des clés d'index dans le tableau PHP n'est pas déterminé par la taille de l'indice, mais par l'ordre du push. Par exemple
$arr[1] = 2; $arr[2] = 3; Pour les clés de type double, Zend HashTable les traitera comme des clés d'index
Zval est une autre structure de données très importante dans zend, utilisée pour identifier et implémenter les variables PHP. Sa structure de données est la suivante :
Zval est principalement composé. de trois Composition de la partie :
La relation correspondante entre les types de variables PHP et leur stockage réel est la suivante :
PHP
1
|
IS_LONG -> lvalueIS_DOUBLE -> dvalueIS_ARRAY -> htIS_STRING -> str IS_RESOURCE ->lvalue |
Le comptage de références est largement utilisé dans le recyclage de la mémoire, les opérations sur les chaînes, etc. Les variables en PHP sont une application typique du comptage de références. Le comptage de références de Zval est implémenté via les variables membres is_ref et ref_count. Grâce au comptage de références, plusieurs variables peuvent partager les mêmes données. Évitez la consommation énorme causée par des copies fréquentes.
Lors de l'exécution d'une opération d'affectation, zend pointe la variable vers les mêmes zval et ref_count++, et pendant l'opération de non-définition, le ref_count-1 correspondant. L'opération de destruction ne sera effectuée que lorsque ref_count sera réduit à 0. S'il s'agit d'une affectation de référence, zend modifiera is_ref à 1.
Les variables PHP réalisent le partage de données variables grâce au comptage de références. Et si vous modifiez la valeur de l'une des variables ? Lorsque vous essayez d'écrire une variable, si Zend constate que le zval pointé par la variable est partagé par plusieurs variables, il copiera un zval avec un ref_count de 1 et décrémentera le refcount du zval d'origine. Ce processus est appelé "séparation zval". ". On peut voir que zend n'effectue des opérations de copie que lorsqu'une opération d'écriture se produit, elle est donc également appelée copie sur écriture (copie sur écriture)
Pour les variables de référence, les exigences sont opposées à celles non- types de référence. Affectation de référence Les variables doivent être regroupées. La modification d'une variable modifie toutes les variables regroupées.
Les nombres entiers et les nombres à virgule flottante sont l'un des types de base en PHP et sont également des variables simples. Pour les entiers et les nombres à virgule flottante, les valeurs correspondantes sont stockées directement dans zvalue. Leurs types sont respectivement longs et doubles.
Comme le montre la structure zvalue, pour les types entiers, contrairement aux langages fortement typés tels que C, PHP ne fait pas de distinction entre les types int, int non signé, long, long long et autres. entiers uniquement Un type est long. De là, on peut voir qu'en PHP, la plage de valeurs des entiers est déterminée par le nombre de bits du compilateur et n'est pas fixe.
Pour les nombres à virgule flottante, similaires aux entiers, il ne fait pas de distinction entre float et double mais seulement double.
En PHP, que dois-je faire si la plage entière est hors limites ? Dans ce cas, il sera automatiquement converti en type double. Vous devez faire attention à cela, car cela provoque de nombreuses astuces.
Comme les entiers, les variables de caractères sont également des types de base et des variables simples en PHP. Comme le montre la structure zvalue, en PHP, une chaîne est composée d'un pointeur vers les données réelles et d'une structure de longueur, similaire à la chaîne en C++. Puisque la longueur est représentée par une variable réelle, contrairement à c, sa chaîne peut être une donnée binaire (incluse). En même temps, en PHP, trouver la longueur de la chaîne strlen est une opération O(1).
Lors de l'ajout, de la modification ou de l'ajout d'opérations de chaîne, PHP réallouera de la mémoire pour générer de nouvelles chaînes. Enfin, pour des raisons de sécurité, PHP ajoutera toujours
à la fin lors de la génération d'une chaîne. Méthodes courantes d'épissage de chaînes et comparaison de vitesse :
Supposons qu'il existe les 4 variables suivantes : $strA= '. 123'; $strB = '456'; $intA=123; intB=456;
Nous allons maintenant comparer et expliquer les méthodes d'épissage de chaînes suivantes :
PH
1 2 3 4 5 6 7 8 |
$res=$strA.$strB和$res=“$strA$strB” 这种情况下,zend会重新malloc一块内存并进行相应处理,其速度一般 $strA=$strA.$strB 这种是速度最快的,zend会在当前strA基础上直接relloc,避免重复拷贝 $res=$intA.$intB 这种速度较慢,因为需要做隐式的格式转换,实际编写程序中也应该注意尽量避免 $strA=sprintf(“%s%s”,$strA.$strB); 这会是最慢的一种方式,因为sprintf在PHP中并不是一个语言结构,本身对于格式识别和处理就需要耗费比较多时间,另外本身机制也是malloc。不过sprintf的方式最具可读性,实际中可以根据具体情况灵活选择。 |
Les tableaux PHP sont naturellement implémentés via Zend HashTable.
Comment implémenter l'opération foreach ? Foreach sur un tableau est complété en parcourant la liste doublement chaînée dans la table de hachage. Pour les tableaux d'index, le parcours via foreach est beaucoup plus efficace que for, éliminant le besoin de rechercher clé->valeur. L’opération de comptage appelle directement l’opération HashTable->NumOfElements, O(1). Pour une chaîne comme « 123 », zend la convertira sous sa forme entière. $arr['123'] et $arr[123] sont équivalents
Les variables de type ressource sont les variables les plus complexes en PHP et constituent également une structure composite.
Le zval de PHP peut représenter un large éventail de types de données, mais il est difficile de décrire entièrement les types de données personnalisés. Puisqu’il n’existe aucun moyen efficace de représenter ces structures composites, il n’existe aucun moyen d’utiliser des opérateurs traditionnels sur celles-ci. Pour résoudre ce problème, il vous suffit de faire référence au pointeur via un identifiant (étiquette) essentiellement arbitraire, appelé ressource.
Dans zval, pour ressource, lval est utilisé comme pointeur, pointant directement vers l'adresse de la ressource. La ressource peut être n'importe quelle structure composite. Les noms familiers mysqli, fsock, memcached, etc. sont tous des ressources.
Comment utiliser les ressources :
Inscription : Pour un type de données personnalisé, vous souhaitez l'utiliser comme ressource. Tout d’abord, vous devez l’enregistrer et Zend lui attribuera un identifiant globalement unique.
Obtenir une variable de ressource : pour les ressources, zend maintient un id->hash_tale des données réelles. Pour une ressource, seul son identifiant est enregistré dans zval. Lors de la récupération, recherchez la valeur spécifique dans hash_table via l'identifiant et renvoyez-la.
Destruction des ressources : les types de données des ressources sont divers. Zend lui-même n'a aucun moyen de le détruire. Par conséquent, les utilisateurs doivent fournir une fonction de destruction lors de l’enregistrement des ressources. Lorsque les ressources sont désactivées, zend appelle la fonction correspondante pour terminer la destruction. Supprimez-le également de la table des ressources globales.
Une ressource peut persister pendant une longue période, non seulement après que toutes les variables y faisant référence soient hors de portée, mais même après la fin d'une requête et une nouvelle requête est effectuée. Ces ressources sont appelées ressources persistantes car elles persistent tout au long du cycle de vie du SAPI, à moins qu'elles ne soient spécifiquement détruites. Dans de nombreux cas, les ressources persistantes peuvent améliorer les performances dans une certaine mesure. Par exemple, dans notre mysql_pconnect commun, les ressources persistantes allouent de la mémoire via pemalloc afin qu'elles ne soient pas libérées à la fin de la requête.
Pour zend, il n'y a pas de distinction entre les deux.
Comment les variables locales et globales sont-elles implémentées en PHP ? Pour une requête, PHP peut voir deux tables de symboles (symbol_table et active_symbol_table) à tout moment, la première étant utilisée pour gérer les variables globales. Ce dernier est un pointeur pointant vers la table de symboles variable actuellement active Lorsque le programme entre dans une fonction, zend lui allouera une table de symboles x et pointera active_symbol_table vers a. De cette manière, la distinction entre les variables globales et locales est réalisée.
Obtenir les valeurs des variables : la table des symboles de PHP est implémentée via hash_table. Chaque variable se voit attribuer un identifiant unique lors de l'obtention, le zval correspondant est trouvé dans la table en fonction de l'identifiant et renvoyé.
Utiliser des variables globales dans les fonctions : Dans les fonctions, nous pouvons utiliser des variables globales en déclarant explicitement global. Créez une référence à la variable du même nom dans symbol_table dans active_symbol_table. S'il n'y a pas de variable du même nom dans symbol_table, elle sera créée en premier.
Le processus d'exécution du mécanisme PHP :
1 Nous n'avons jamais démarré manuellement le processus lié à PHP, il s'exécute au démarrage d'Apache
2. vers Apache via le module mod_php5.so (plus précisément SAPI, l'interface de programmation d'applications serveur
3. PHP a un total de trois modules : le noyau, le moteur Zend et la couche d'extension ;
. 4. Le noyau PHP est utilisé pour gérer les requêtes, les flux de fichiers, la gestion des erreurs et autres opérations associées
5 Le moteur Zend (ZE) est utilisé pour convertir les fichiers sources en langage machine puis s'exécuter sur le virtuel ; machine It;
6. La couche d'extension est un ensemble de fonctions, de bibliothèques et de flux que PHP utilise pour effectuer certaines opérations spécifiques. Par exemple, nous avons besoin de l'extension mysql pour nous connecter à la base de données MySQL ;
7. Lorsque ZE exécute le programme, il peut avoir besoin de se connecter à plusieurs extensions. À ce stade, ZE cède le contrôle à l'extension. et renvoyez-le après avoir traité la tâche spécifique ;
8. Enfin, ZE renvoie les résultats de l'exécution du programme au noyau PHP, qui transmet ensuite les résultats à la couche SAPI et les envoie enfin au navigateur.
Discussion approfondie du mécanisme d'exploitation PHP
La première étape du démarrage PHP du mécanisme d'exploitation PHP
Vous ne savez pas quelles sont la première et la deuxième étapes ? Ne vous inquiétez pas, nous en discuterons en détail ensuite.
Jetons un coup d’œil à la première et la plus importante étape. Ce qu’il faut retenir, c’est que la première étape de l’opération se déroule avant l’arrivée des requêtes. Après avoir démarré Apache, l'interpréteur PHP démarre également ; PHP appelle la méthode MINIT de chaque extension, faisant ainsi passer ces extensions dans un état disponible. Jetez un œil aux extensions ouvertes dans le fichier php.ini ; MINIT signifie "initialisation du module". Chaque module définit un ensemble de fonctions, de bibliothèques de classes, etc. pour gérer d'autres requêtes.
Une méthode MINIT typique est la suivante :
PHP_MINIT_FUNCTION(extension_name){ /* Initialiser les fonctions, les classes, etc. */ }
La deuxième étape du démarrage PHP du mécanisme d'exploitation PHP
Lorsqu'une demande de page se produit, la couche SAPI cède le contrôle à la couche PHP. PHP définit donc les variables d'environnement nécessaires pour répondre à cette requête. Dans le même temps, il crée également une table de variables pour stocker les noms de variables et les valeurs générées lors de l'exécution. PHP appelle la méthode RINIT de chaque module, qui est la « requête d'initialisation ». Un exemple classique est le RINIT du module Session. Si le module Session est activé dans php.ini, la variable $_SESSION sera initialisée et le contenu pertinent sera lu lorsque le RINIT du module sera appelé ; être considéré comme un Le processus de préparation démarre automatiquement entre les exécutions du programme.
Une méthode RINIT typique est la suivante :
PHP_RINIT_FUNCTION(extension_name) { /* Initialiser les variables de session, pré-remplir les variables, redéfinir les variables globales, etc. */ }
PHP run La première étape du mécanisme d'arrêt de PHP
Tout comme le démarrage de PHP, l'arrêt de PHP est également divisé en deux étapes : Une fois la page exécutée (qu'elle atteigne la fin du fichier ou qu'elle se termine avec la sortie ou la mort fonction), PHP lancera la procédure de nettoyage. Il appellera la méthode RSHUTDOWN de chaque module dans l'ordre. RSHUTDOWN permet d'effacer la table des symboles générée lors de l'exécution du programme, c'est-à-dire d'appeler la fonction unset sur chaque variable.
Une méthode RSHUTDOWN typique est la suivante :
PHP_RSHUTDOWN_FUNCTION(extension_name) { /* Effectuer la gestion de la mémoire, supprimer toutes les variables utilisées lors du dernier appel PHP, etc. */ }
PHP La deuxième étape d'arrêt de PHP du mécanisme d'exploitation
Enfin, toutes les requêtes ont été traitées et SAPI est prêt à être arrêté. PHP commence à exécuter la deuxième étape : PHP appelle la méthode MSHUTDOWN de chaque extension. C'est le module. Dernière chance de libérer de la mémoire.
Une méthode RSHUTDOWN typique est la suivante :
PHP_MSHUTDOWN_FUNCTION(extension_name) { /* Gestionnaires gratuits et mémoire persistante, etc. */ }
De cette façon, toute la vie de PHP le cycle est terminé. A noter que « démarrer la première étape » et « fermer la deuxième étape » ne seront exécutés que lorsqu'il n'y a pas de requête du serveur.
Recommandations associées :
Explication détaillée du code du mécanisme de fonctionnement sous-jacent des fermetures JavaScript
Le mécanisme de fonctionnement sous-jacent des fermetures JavaScript
Exploration du mécanisme de fonctionnement sous-jacent du tutoriel PHP_PHP
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!