Einige Einführungen in den Abstract Syntax Tree (AST) in den neuen Funktionen von PHP7

不言
Freigeben: 2023-04-03 10:22:02
Original
2151 Leute haben es durchsucht

Dieser Artikel gibt Ihnen eine Einführung in die neuen Funktionen von PHP7. Ich hoffe, dass er allen helfen kann.

Dieser Artikel analysiert die Änderungen, die durch die neue Funktion von PHP7, Abstract Syntax Tree (AST), hervorgerufen werden. Teilen Sie es als Referenz mit allen:

Der Großteil des Inhalts basiert auf dem RFC-Dokument von AST: https://wiki.php.net/rfc/abstractsyntaxtree Zum Verständnis wird ein Auszug aus dem Quelldokument eingeführt.

In diesem Artikel erfahren Sie nicht, was ein abstrakter Syntaxbaum ist. Sie müssen ihn selbst verstehen. Dieser Artikel beschreibt nur einige Änderungen, die AST an PHP mit sich bringt.

Neuer Ausführungsprozess

Eine wichtige Änderung im Kern von PHP7 ist die Hinzufügung von AST. In PHP5 ist der Ausführungsprozess von PHP-Skripten zu Opcodes:

  1. Lexing: lexikalische Scan-Analyse, Konvertieren von Quelldateien in Token-Streams

  2. Parsing: Syntaxanalyse, in dieser Phase werden Op-Arrays generiert.

In PHP7 werden Op-Arrays nicht mehr direkt während der Syntaxanalysephase generiert, sondern AST wird zuerst generiert, daher gibt es einen weiteren Schritt im Prozess:

  1. Lexing: Lexikalische Scan-Analyse, Konvertierung von Quelldateien in Token-Streams;

  2. Parsing: Syntax-Analyse, Generierung abstrakter Syntaxbäume aus Token-Streams;

  3. Kompilierung: Generieren Sie Op-Arrays aus abstrakten Syntaxbäumen.
  4. Ausführungszeit und Speicherverbrauch

Von den oben genannten Schritten ist dies ein Schritt mehr als der vorherige Prozess, sodass dies nach gesundem Menschenverstand die Programmausführung erhöht Zeit- und Speicherverbrauch. Tatsächlich hat sich jedoch die Speichernutzung erhöht, die Ausführungszeit jedoch abgenommen.

Die folgenden Ergebnisse werden durch das Testen von drei Skripten erzielt: klein (ca. 100 Zeilen Code), mittel (ca. 700 Zeilen) und groß (ca. 2800 Zeilen). Testskript: https://gist.github .com/nikic/289b0c7538b46c2220bc.

Ausführungszeit für das 100-fache Kompilieren jeder Datei (beachten Sie, dass das Testergebnis des Artikels 14 Jahre zurückliegt, als PHP7 noch PHP-NG hieß):

Speicherspitze in einer einzelnen Kompilierung:

php-ng php-ast diff
SMALL 0.180s 0.160s -12.5%
MEDIUM 1.492s 1.268s -17.7%
LARGE 6.703s 5.736s -16.9%

Die Testergebnisse einer einzelnen Kompilierung spiegeln möglicherweise nicht die tatsächliche Nutzung wider. Im Folgenden sind die Ergebnisse eines vollständigen Projekttests mit PhpParser aufgeführt:

php-ng php-ast diff
SMALL 378kB 414kB +9.5%
MEDIUM 507kB 643kB +26.8%
LARGE 1084kB 1857kB +71.3%

Tests zeigen, dass sich nach der Verwendung von AST die Gesamtausführungszeit des Programms um etwa 10 bis 15 % verbessert, der Speicherverbrauch jedoch auch zunimmt. Der Anstieg ist bei einer einzelnen Zusammenstellung großer Dateien offensichtlich, nicht jedoch bei gesamter Projektausführungsprozess. Sehr ernstes Problem.

Beachten Sie außerdem, dass die oben genannten Ergebnisse alle ohne Opcache gelten. Wenn Opcache in einer Produktionsumgebung aktiviert ist, stellt der Anstieg des Speicherverbrauchs kein großes Problem dar.

Semantische Änderungen

Wenn es sich nur um eine Zeitoptimierung handelt, scheint dies kein ausreichender Grund für die Verwendung von AST zu sein. Tatsächlich basiert die Implementierung von AST nicht auf Überlegungen zur Zeitoptimierung, sondern auf der Lösung von Syntaxproblemen. Werfen wir einen Blick auf einige Änderungen in der Semantik.

Yield erfordert keine Klammern

Wenn Sie in der PHP5-Implementierung yield in einem Ausdruckskontext verwenden (z. B. auf der rechten Seite eines Zuweisungsausdrucks), müssen Sie Verwenden Sie Klammern auf beiden Seiten der Anweisung: yield

<?php
$result = yield fn(); // 不合法的
$result = (yield fn()); // 合法的
Nach dem Login kopieren

Dieses Verhalten ist nur auf die Implementierungsbeschränkungen von PHP5 zurückzuführen. In PHP7 sind Klammern nicht mehr erforderlich. Daher sind auch folgende Schreibmethoden zulässig:

<?php
$result = yield;
$result = yield $v;
$result = yield $k => $v;
Nach dem Login kopieren

Natürlich müssen Sie sich an die Anwendungsszenarien von

halten. yield

Klammern haben keinen Einfluss auf das Verhalten

In PHP5 sind die Bedeutungen der beiden Anweisungen

und ($foo)[&#39;bar&#39;] = &#39;baz&#39; unterschiedlich. Tatsächlich ist die erste Schreibweise illegal und Sie erhalten die folgende Fehlermeldung: $foo[&#39;bar&#39;] = &#39;baz&#39;

<?php
($foo)[&#39;bar&#39;] = &#39;baz&#39;;
# PHP Parse error: Syntax error, unexpected &#39;[&#39; on line 1
Nach dem Login kopieren

Aber in PHP7 bedeuten die beiden Schreibweisen dasselbe.

Wenn die Parameter der Funktion in Klammern stehen, gibt es ebenfalls ein Problem mit der Typprüfung. Dieses Problem wurde auch in PHP7 gelöst:

<?php
function func() {
 return [];
}
function byRef(array &$a) {
}
byRef((func()));
Nach dem Login kopieren

Der obige Code wird nicht angezeigt PHP5, es sei denn, es wird

verwendet, aber in PHP7 tritt unabhängig davon, ob auf beiden Seiten von byRef(func()) Klammern stehen, der folgende Fehler auf: func()

PHP Strenge Standards: Nur Variablen sollten als Referenz übergeben werden .. .

Änderungen an list()

Das Verhalten des Listenschlüsselworts hat sich stark geändert. Die Reihenfolge, in der die Liste den Variablen Werte zuweist (die Reihenfolge links und rechts des Gleichheitszeichens gleichzeitig), war früher von rechts nach links, jetzt jedoch von links nach rechts:

<?php
list($array[], $array[], $array[]) = [1, 2, 3];
var_dump($array);
// PHP5: $array = [3, 2, 1]
// PHP7: $array = [1, 2, 3]
# 注意这里的左右的顺序指的是等号左右同时的顺序,
# list($a, $b) = [1, 2] 这种使用中 $a == 1, $b == 2 是没有疑问的。
Nach dem Login kopieren

Der Grund für die obige Änderung liegt genau darin, dass im Zuweisungsprozess von PHP5 3 zuerst und 1 zuletzt in das Array eingefügt würden, aber jetzt hat sich die Reihenfolge geändert.

Die gleichen Änderungen sind:

<?php
$a = [1, 2];
list($a, $b) = $a;
// PHP5: $a = 1, $b = 2
// PHP7: $a = 1, $b = null + "Undefined index 1"
Nach dem Login kopieren

Das liegt daran, dass $b im vorherigen Zuweisungsprozess zuerst 2 bekam und dann der Wert von $a 1 wurde, aber jetzt ändert sich $a zuerst wird 1 und ist kein Array mehr, sodass $b null wird.

Auf die Liste wird jetzt nur einmal pro Offset zugegriffen:

<?php
list(list($a, $b)) = $array;
// PHP5:
$b = $array[0][1];
$a = $array[0][0];
// PHP7:
// 会产生一个中间变量,得到 $array[0] 的值
$_tmp = $array[0];
$a = $_tmp[0];
$b = $_tmp[1];
Nach dem Login kopieren

Leere Listenmitglieder sind jetzt vollständig verboten, bisher nur in bestimmten Fällen:

<?php
list() = $a;   // 不合法
list($b, list()) = $a; // 不合法
foreach ($a as list()) // 不合法 (PHP5 中也不合法)
Nach dem Login kopieren

Die Reihenfolge der Referenzzuweisung

Die Reihenfolge der Referenzzuweisung ist in PHP5 von rechts nach links und im Präsens von links nach rechts:

<?php
$obj = new stdClass;
$obj->a = &$obj->b;
$obj->b = 1;
var_dump($obj);
// PHP5:
object(stdClass)#1 (2) {
 ["b"] => &int(1)
 ["a"] => &int(1)
}
// PHP7:
object(stdClass)#1 (2) {
 ["a"] => &int(1)
 ["b"] => &int(1)
}
Nach dem Login kopieren

__clone-Methode kann direkt aufgerufen werden

Jetzt können Sie direkt Verwenden Sie die Schreibmethode

, um die Methode $obj->__clone() aufzurufen. __clone ist die einzige magische Methode, deren direkter Aufruf zuvor verboten war: __clone

Schwerwiegender Fehler: Methode __clone() kann nicht für Objekte aufgerufen werden – verwenden Sie „clone $obj“. stattdessen in ...

Variablensyntaxkonsistenz

AST löst auch einige Syntaxkonsistenzprobleme, die in einem anderen RFC aufgeworfen wurden: https://wiki.php net/rfc/uniform_variable_syntax.

In der neuen Implementierung ist die Bedeutung einiger früherer grammatikalischer Ausdrücke etwas anders als jetzt. Einzelheiten finden Sie in der folgenden Tabelle:

Expression PHP5 PHP7
$$foo['bar']['baz'] ${$foo['bar']['baz']} ($$foo)['bar']['baz']
$foo->$bar['baz'] $foo->{$bar['baz']} ($foo->$bar)['baz']
$foo->$bar['baz']() $foo->{$bar['baz']}() ($foo->$bar)['baz']()
Foo::$bar['baz']() Foo::{$bar['baz']}() (Foo::$bar)['baz']()
Insgesamt ist es immer noch dasselbe Bisher war die Reihenfolge von rechts nach links, jetzt von links nach rechts, wobei auch der Grundsatz befolgt wurde, dass Klammern das Verhalten nicht beeinflussen. Diesen komplexen Methoden zum Schreiben variabler Variablen muss in der tatsächlichen Entwicklung Beachtung geschenkt werden.

Verwandte Empfehlungen:

Fünf Möglichkeiten, Ihnen beizubringen, wie Sie PHP-Fehlerechoinformationen deaktivieren können

FastCGI-Protokoll in PHP Quellcode-Analyse von

Das obige ist der detaillierte Inhalt vonEinige Einführungen in den Abstract Syntax Tree (AST) in den neuen Funktionen von PHP7. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Verwandte Etiketten:
Quelle:php.cn
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage
Über uns Haftungsausschluss Sitemap
Chinesische PHP-Website:Online-PHP-Schulung für das Gemeinwohl,Helfen Sie PHP-Lernenden, sich schnell weiterzuentwickeln!