Cet article résume les points de connaissances pertinents sur l'analyse et l'application des liaisons statiques avancées de PHP. Les amis qui sont intéressés par cela peuvent l'apprendre.
Connaissances de base
1. Opérateur d'analyse de plage (::)
peut être utilisé pour accéder aux membres statiques, aux constantes de classe et peut également être utilisé pour remplacer les propriétés et les méthodes d'une classe.
Les trois mots-clés spéciaux self, parent et static sont utilisés pour accéder à ses propriétés ou méthodes à l'intérieur de la définition de classe.
parent est utilisé pour appeler des propriétés ou des méthodes remplacées dans la classe parent (là où il apparaît, il sera résolu en classe parent de la classe correspondante).
self est utilisé pour appeler des méthodes ou des propriétés dans cette classe (partout où il apparaît, il sera analysé dans la classe correspondante ; notez la différence avec $this, $this pointe vers l'actuel objet instancié).
Lorsqu'une sous-classe substitue une méthode dans sa classe parent, PHP n'appellera pas la méthode substituée dans la classe parent. L’appel de la méthode de la classe parent dépend de la classe enfant.
2. Le noyau PHP place l'implémentation de l'héritage des classes dans la "phase de compilation"
<?php class A{ const H = 'A'; const J = 'A'; static function testSelf(){ echo self::H; //在编译阶段就确定了 self解析为 A } } class B extends A{ const H = "B"; const J = 'B'; static function testParent(){ echo parent::J; //在编译阶段就确定了 parent解析为A } /* 若重写testSelf则能输出“B”, 且C::testSelf()也是输出“B” static function testSelf(){ echo self::H; } */ } class C extends B{ const H = "C"; const J = 'C'; } B::testParent(); B::testSelf(); echo "\n"; C::testParent(); C::testSelf();
Résultat de l'exécution :
AA
AA
Conclusion :
soi : : et parent:: apparaissent dans la définition d'une classe X, ils seront analysés dans la classe X correspondante, à moins que la méthode de la classe parent ne soit remplacée dans la sous-classe.
3.Mot-clé statique (statique)
Fonction :
- Le mot-clé statique est utilisé pour modifier les variables dans le corps de la fonction Définir variables locales statiques.
- Utilisé pour déclarer des membres statiques lors de la modification des fonctions membres de la classe et des variables membres.
- (après PHP5.3) une classe spéciale qui représente la liaison statique retardée avant le résolveur de portée (::).
Exemple :
Définir des variables locales statiques (occurrence : dans les fonctions locales)
Fonctionnalité : Les variables statiques n'existent que dans le domaine de la fonction locale, mais lorsque le programme Sa valeur n'est pas perdue lorsque l'exécution sort de ce cadre.
<?php function test() { static $count = 0; $count++; echo $count; if ($count < 10) { test(); } $count--; }
Définir des méthodes statiques, des attributs statiques
a) Déclarez les attributs de classe ou les méthodes comme statiques, vous ne le faites pas il faut les instancier. Classification et accès direct.
b) Les propriétés statiques ne sont pas accessibles via un objet dont la classe a été instanciée (mais les méthodes statiques le peuvent)
c) Si aucun contrôle d'accès n'est spécifié, les propriétés et méthodes La valeur par défaut est publique.
d) Étant donné que les méthodes statiques ne nécessitent pas l'appel d'un objet, la pseudo variable $this n'est pas disponible dans les méthodes statiques.
e) Les propriétés statiques ne sont pas accessibles aux objets via l'opérateur ->
f) L'appel statique d'une méthode non statique entraînera une erreur de niveau E_STRICT.
g) Comme toutes les autres variables statiques PHP, les propriétés statiques ne peuvent être initialisées qu'avec des littéraux ou des constantes, pas des expressions. Ainsi, une propriété statique peut être initialisée avec un entier ou un tableau, mais elle ne peut pas être initialisée avec une autre valeur de retour de variable ou de fonction, ni pointer vers un objet.
a. Exemple de méthode statique (emplacement d'apparition : définition de la méthode de classe)
<?php class Foo { public static function aStaticMethod() { // ... } } Foo::aStaticMethod(); $classname = 'Foo'; $classname::aStaticMethod(); // 自PHP 5.3.0后,可以通过变量引用类 ?>
b. définition de l'attribut)
<?php class Foo { public static $my_static = 'foo'; public function staticValue() { return self::$my_static; //self 即 FOO类 } } class Bar extends Foo { public function fooStatic() { return parent::$my_static; //parent 即 FOO类 } } print Foo::$my_static . "\n"; $foo = new Foo(); print $foo->staticValue() . "\n"; print $foo->my_static . "\n"; // Undefined "Property" my_static print $foo::$my_static . "\n"; $classname = 'Foo'; print $classname::$my_static . "\n"; // As of PHP 5.3.0 print Bar::$my_static . "\n"; $bar = new Bar(); print $bar->fooStatic() . "\n"; ?>
c. Utilisé pour la liaison statique tardive (position : dans la méthode de la classe, utilisée pour modifier des variables ou des méthodes)
Ce qui suit est une analyse détaillée de
liaison statique tardive (liaison statique tardive)
Depuis PHP 5.3.0, PHP Ajout d'une fonctionnalité appelée liaison statique tardive pour référencer des classes appelées statiquement dans la portée de l'héritage.
1. Appels renvoyés et appels non renvoyés
Appels renvoyés :
fait référence aux appels statiques passés de la manière suivante : soi ::, parent ::, static :: et forward_static_call().
Appels non transférés :
Appels statiques qui spécifient explicitement le nom de la classe (par exemple Foo::foo())
Appels non statiques (par exemple $ foo->foo())
2. Principe de fonctionnement de la liaison statique tardive
Principe : stocker le "non" précédent -forwarded" Nom de classe dans un appel sans transfert. Cela signifie que lorsque nous appelons un appel statique qui est un appel direct, la classe réellement appelée est la classe de l'appel non direct précédent.
Exemple d'analyse :
<?php class A { public static function foo() { echo __CLASS__."\n"; static::who(); } public static function who() { echo __CLASS__."\n"; } } class B extends A { public static function test() { echo "A::foo()\n"; A::foo(); echo "parent::foo()\n"; parent::foo(); echo "self::foo()\n"; self::foo(); } public static function who() { echo __CLASS__."\n"; } } class C extends B { public static function who() { echo __CLASS__."\n"; } } C::test(); /* * C::test(); //非转发调用 ,进入test()调用后,“上一次非转发调用”存储的类名为C * * //当前的“上一次非转发调用”存储的类名为C * public static function test() { * A::foo(); //非转发调用, 进入foo()调用后,“上一次非转发调用”存储的类名为A,然后实际执行代码A::foo(), 转 0-0 * parent::foo(); //转发调用, 进入foo()调用后,“上一次非转发调用”存储的类名为C, 此处的parent解析为A ,转1-0 * self::foo(); //转发调用, 进入foo()调用后,“上一次非转发调用”存储的类名为C, 此处self解析为B, 转2-0 * } * * * 0-0 * //当前的“上一次非转发调用”存储的类名为A * public static function foo() { * static::who(); //转发调用, 因为当前的“上一次非转发调用”存储的类名为A, 故实际执行代码A::who(),即static代表A,进入who()调用后,“上一次非转发调用”存储的类名依然为A,因此打印 “A” * } * * 1-0 * //当前的“上一次非转发调用”存储的类名为C * public static function foo() { * static::who(); //转发调用, 因为当前的“上一次非转发调用”存储的类名为C, 故实际执行代码C::who(),即static代表C,进入who()调用后,“上一次非转发调用”存储的类名依然为C,因此打印 “C” * } * * 2-0 * //当前的“上一次非转发调用”存储的类名为C * public static function foo() { * static::who(); //转发调用, 因为当前的“上一次非转发调用”存储的类名为C, 故实际执行代码C::who(),即static代表C,进入who()调用后,“上一次非转发调用”存储的类名依然为C,因此打印 “C” * } */ 故最终结果为: A::foo() A A parent::foo() A C self::foo() A C
3. Plus d'exemples de liaison statique tardive
a) Comparaison de soi, du parent et de la statique
<?php class Mango { function classname(){ return __CLASS__; } function selfname(){ return self::classname(); } function staticname(){ return static::classname(); } } class Orange extends Mango { function parentname(){ return parent::classname(); } function classname(){ return __CLASS__; } } class Apple extends Orange { function parentname(){ return parent::classname(); } function classname(){ return __CLASS__; } } $apple = new Apple(); echo $apple->selfname() . "\n"; echo $apple->parentname() . "\n"; echo $apple->staticname(); ?> 运行结果: Mango Orange Apple
b) Utilisation de forward_static_call()
<?php class Mango { const NAME = 'Mango is'; public static function fruit() { $args = func_get_args(); echo static::NAME, " " . join(' ', $args) . "\n"; } } class Orange extends Mango { const NAME = 'Orange is'; public static function fruit() { echo self::NAME, "\n"; forward_static_call(array('Mango', 'fruit'), 'my', 'favorite', 'fruit'); forward_static_call('fruit', 'my', 'father\'s', 'favorite', 'fruit'); } } Orange::fruit('NO'); function fruit() { $args = func_get_args(); echo "Apple is " . join(' ', $args). "\n"; } ?> 运行结果: Orange is Orange is my favorite fruit Apple is my father's favorite fruit
c) Utilisez get_call_class()
<?php class Mango { static public function fruit() { echo get_called_class() . "\n"; } } class Orange extends Mango { // } Mango::fruit(); Orange::fruit(); ?> 运行结果: Mango Orange
Appliquer
Comme mentionné précédemment, le but de l'introduction d'une liaison statique tardive est de référencer des classes appelées statiquement dans la portée de l'héritage.
Vous pouvez donc utiliser une liaison statique tardive pour résoudre le problème de l'héritage singleton.
Regardons d'abord ce que signifie utiliser self :
<?php // new self 得到的单例都为A。 class A { protected static $_instance = null; protected function __construct() { //disallow new instance } protected function __clone(){ //disallow clone } static public function getInstance() { if (self::$_instance === null) { self::$_instance = new self(); } return self::$_instance; } } class B extends A { protected static $_instance = null; } class C extends A{ protected static $_instance = null; } $a = A::getInstance(); $b = B::getInstance(); $c = C::getInstance(); var_dump($a); var_dump($b); var_dump($c); 运行结果: E:\code\php_test\apply\self.php:37: class A#1 (0) { } E:\code\php_test\apply\self.php:38: class A#1 (0) { } E:\code\php_test\apply\self.php:39: class A#1 (0) { }
Comme vous pouvez le voir dans l'exemple ci-dessus, utiliser self , Tous les objets instanciés sont le même objet de classe A
Voyons quels résultats seront obtenus en utilisant statique
<?php // new static 得到的单例分别为D,E和F。 class D { protected static $_instance = null; protected function __construct(){} protected function __clone() { //disallow clone } static public function getInstance() { if (static::$_instance === null) { static::$_instance = new static(); } return static::$_instance; } } class E extends D { protected static $_instance = null; } class F extends D{ protected static $_instance = null; } $d = D::getInstance(); $e = E::getInstance(); $f = F::getInstance(); var_dump($d); var_dump($e); var_dump($f); 运行结果: E:\code\php_test\apply\static.php:35: class D#1 (0) { } E:\code\php_test\apply\static.php:36: class E#2 (0) { } E:\code\php_test\apply\static.php:37: class F#3 (0) { }
Ce qui précède c'est ce que j'ai compilé pour tout le monde. J'espère que cela sera utile à tout le monde à l'avenir.
Articles connexes :
AjaxSubmit() soumet le fichier
Deux méthodes permettant à Ajax de résoudre les actualisations redondantes
Une brève analyse et des solutions à la synchronisation Ajax et aux problèmes asynchrones
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!