이 기사에서는 PHP7의 새로운 기능인 추상 구문 트리(AST)에 대해 소개합니다. 내용이 매우 좋습니다. 도움이 필요한 모든 사람에게 도움이 되기를 바랍니다.
이 글에서는 PHP7의 새로운 기능인 AST(Abstract Syntax Tree)로 인한 변화를 분석합니다. 참조를 위해 모든 사람과 공유하세요. 세부 사항은 다음과 같습니다:
여기에 있는 대부분의 콘텐츠는 AST의 RFC 문서인 https://wiki.php.net/rfc/abstractsyntaxtree를 참조합니다. 이해를 돕기 위해 원본 문서에서 다음과 같은 내용을 발췌하여 소개합니다.
이 기사에서는 추상 구문 트리가 무엇인지 설명하지 않습니다. 이 기사에서는 AST가 PHP에 가져오는 몇 가지 변경 사항만 설명합니다.
PHP7 커널의 중요한 변화는 AST의 추가입니다. PHP5에서 PHP 스크립트에서 opcode까지의 실행 프로세스는 다음과 같습니다.
Lexing: 어휘 스캐닝 분석, 소스 파일을 토큰 스트림으로 변환
# 🎜🎜##🎜 🎜## 🎜🎜#
Lexing: 어휘 스캐닝 분석, 소스 파일을 토큰 스트림으로 변환
php-ast | # ㅋㅋㅋ 🎜# - 12.5% MEDIUM | 1.492s | |
-1 7.7% 에디토리얼 #🎜 🎜# | 단일 편집의 메모리 최고치: | ||
# 🎜 🎜# | php-ng | php-ast | diff |
SMALL#🎜🎜 # | 378kB414kB | +9.5% |
1857kB
php-ast | diff | ||||||||||||||||
25.5ms | 22.8ms | -11.8% | #🎜 🎜# | MEMORY||||||||||||||
# 🎜🎜 #2482kB | +5.1% | #🎜 🎜##🎜 🎜# 테스트 결과, AST를 사용한 후 프로그램의 전체 실행 시간이 약 10%~15% 향상되었지만 메모리 소비도 증가하는 것으로 나타났습니다. 대용량 파일을 한 번 컴파일할 때 그 증가는 명백하지만 그렇습니다. 전체 프로젝트 실행 과정에서 증가하지 않습니다. 심각한 문제는 아닙니다. 또한 위의 결과는 모두 Opcache를 사용하지 않은 결과이므로 프로덕션 환경에서 Opcache를 켜면 메모리 소비 증가는 큰 문제가 되지 않습니다. 의미적 변경단순한 시간 최적화라면 AST를 사용할 이유가 충분하지 않은 것 같습니다. 실제로 AST 구현은 시간 최적화 고려사항이 아니라 구문 문제를 해결하기 위한 것입니다. 의미론의 몇 가지 변화를 살펴보겠습니다. yield에는 대괄호가 필요하지 않습니다.PHP5 구현에서 <?php $result = yield fn(); // 不合法的 $result = (yield fn()); // 合法的 这种行为仅仅是因为 PHP5 的实现方式的限制,在 PHP7 中,括号不再是必须的了。所以下面这些写法也都是合法的: <?php $result = yield; $result = yield $v; $result = yield $k => $v; 当然了,还得遵循 括号不影响行为在 PHP5 中, <?php ($foo)['bar'] = 'baz'; # PHP Parse error: Syntax error, unexpected '[' on line 1 但是在 PHP7 中,两种写法表示同样的意思。 同样,如果函数的参数被括号包裹,类型检查存在问题,在 PHP7 中这个问题也得到了解决: <?php function func() { return []; } function byRef(array &$a) { } byRef((func())); 以上代码在 PHP5 中不会告警,除非使用 PHP Strict standards: Only variables should be passed by reference ... list() 的变化list 关键字的行为改变了很多。list 给变量赋值的顺序(等号左右同时的顺序)以前是从右至左,现在是从左到右: <?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 是没有疑问的。 产生上面变化的原因正是因为在 PHP5 的赋值过程中,3 会最先被填入数组,1 最后,但是现在顺序改变了。 同样的变化还有: <?php $a = [1, 2]; list($a, $b) = $a; // PHP5: $a = 1, $b = 2 // PHP7: $a = 1, $b = null + "Undefined index 1" 这是因为在以前的赋值过程中 $b 先得到 2,然后 $a 的值才变成 1,但是现在 $a 先变成了 1,不再是数组,所以 $b 就成了 null。 list 现在只会访问每个偏移量一次: <?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]; 空的 list 成员现在是全部禁止的,以前只是在某些情况下: <?php list() = $a; // 不合法 list($b, list()) = $a; // 不合法 foreach ($a as list()) // 不合法 (PHP5 中也不合法) 引用赋值的顺序引用赋值的顺序在 PHP5 中是从右到左的,现在时从左到右: <?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) } __clone 方法可以直接调用现在可以直接使用 rrreee 물론yield 의 적용 시나리오를 따라야 합니다. 괄호는 동작에 영향을 주지 않습니다PHP5에서는 ($foo)['bar'] = 'baz' 및 $foo ['bar'] = 'baz' 두 문은 서로 다른 의미를 갖습니다. 사실, 이전 작성 방법은 불법이며 다음과 같은 오류가 발생합니다: rrreee 그러나 PHP7에서는 두 가지 작성 방법이 동일한 의미를 갖습니다.
# 🎜🎜# $foo->$bar['baz']()$foo->{$bar['baz']}() #🎜🎜#( $foo- >$bar)['baz']()#🎜🎜##🎜🎜##🎜🎜##🎜🎜#Foo::$bar['baz']()#🎜🎜##🎜🎜 #Foo: :{$bar['baz']}()#🎜🎜##🎜🎜#(Foo::$bar)['baz']()#🎜🎜##🎜🎜##🎜🎜## 🎜🎜# #🎜🎜# 전체적으로 이전 순서는 오른쪽에서 왼쪽이었지만 이제는 왼쪽에서 오른쪽이며 괄호가 동작에 영향을 주지 않는다는 원칙도 따릅니다. 이러한 복잡한 변수 작성 방식은 실제 개발 시 주의가 필요합니다. #🎜🎜##🎜🎜#관련 권장 사항: #🎜🎜##🎜🎜##🎜🎜#PHP 오류 에코 정보를 끄는 방법을 가르치는 5가지 방법#🎜🎜##🎜🎜##🎜🎜##🎜 🎜 ##🎜🎜#PHP 소스 코드의 FastCGI 프로토콜 분석#🎜🎜##🎜🎜##🎜🎜##🎜🎜##🎜🎜##🎜🎜# |
위 내용은 PHP7의 새로운 기능 중 추상 구문 트리(AST)에 대한 일부 소개의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!