새로운 주요 PHP 버전인 PHP8은 2020년 말에 출시될 예정입니다.
현재 매우 활발한 개발이 진행 중이므로 앞으로 몇 달에 걸쳐 개발 속도와 진행 상황이 많이 바뀔 수 있습니다.
이 글에서는 PHP8에서 일어날 몇 가지 변경 사항, 즉 새로운 기능, 성능 개선, 주요 변경 사항을 나열하겠습니다.
PHP8은 새로운 메이저 버전이기 때문에 코드와 구문의 하위 호환성이 낮습니다.
최신 버전을 유지하고 있다면 대부분의 주요 변경 사항이 7.* 버전에서 더 이상 사용되지 않으므로 업그레이드가 그리 어렵지 않을 것입니다.
획기적인 변화 외에도 PHP8은 JIT 컴파일러 및 공용체 유형과 같은 몇 가지 훌륭한 새 기능을 제공하며 물론 더 많은 기능이 있습니다.
새로운 기능
새로운 기능으로 시작하세요. 하지만 PHP8은 아직 개발 중이므로 이 목록은 시간이 지나면서 늘어날 것입니다.
Union 유형 RFC
PHP의 동적 타이핑 특성을 고려할 때, Union 유형은 여러 상황에서 유용합니다.
합집합 유형은 두 유형 중 하나를 사용할 수 있음을 나타내는 두 개 이상의 유형 모음입니다.
public function foo(Foo|Bar $input): int|float;
왜 이게 C언어의 Union과 좀 비슷하다고 느껴지는 걸까요?
void는 "반환 값이 전혀 없음"을 의미하므로 공용체 유형의 일부가 될 수 없습니다.
또한 |NULL을 사용하거나 기존 항목을 사용할 수 있나요? .
public function foo(Foo|null $foo): void; public function bar(?Bar $bar): void;
JIT RFC
JIT-Just-In-Time 컴파일러는 상당한 성능 향상을 약속하지만 웹 애플리케이션에서는 큰 이점이 없을 수도 있습니다.
현재 정확한 벤치마크는 없지만, 분명히 나올 것입니다.
정적 반환 유형(Static return type) RFC
self 반환은 가능하지만 PHP8 이전에는 static이 유효한 반환 유형이 아니었습니다. PHP는 동적으로 유형이 지정되는 특성을 고려하는 많은 개발자에게 유용합니다.
class Foo { public function test(): static { return new static(); } }
Weak Maps RFC
PHP 7.4에 추가된 WeakRefs RFC를 기반으로 WeakMap 구현이 PHP 8에 추가되었습니다. WeakMap에는 객체에 대한 참조가 포함되어 있어 이러한 객체가 가비지 수집되는 것을 방지하지 않습니다.
ORM을 예로 들면, ORM은 엔터티 간의 관계 성능을 향상시키기 위해 엔터티 클래스에 대한 참조가 포함된 캐시를 구현하는 경우가 많습니다.
이러한 엔터티 개체는 캐시가 개체를 참조하는 유일한 개체인 경우에도 캐시에 참조가 있는 한 가비지 수집될 수 없습니다.
이 캐싱 계층이 약한 참조와 매핑을 대신 사용하는 경우 PHP는 다른 객체에서 더 이상 참조하지 않는 객체를 가비지 수집합니다.
특히 단일 요청으로 수백(수천은 아니더라도) 엔터티를 관리할 수 있는 ORM의 경우 약한 매핑은 이러한 객체를 처리하는 더 좋고 리소스 친화적인 방법을 제공할 수 있습니다.
RFC의 예인 약한 맵의 사용법은 다음과 같습니다.
class Foo { private WeakMap $cache; public function getSomethingWithCaching(object $obj): object { return $this->cache[$obj] ??= $this->computeSomethingExpensive($obj); } }
객체에 ::class를 사용할 수 있습니다. RFC
작지만 유용한 새 기능: 이제 객체를 사용하지 않고도 객체에 ::class를 사용할 수 있습니다. get_class()를 사용하세요.
get_class()와 동일하게 작동합니다.
$foo = new Foo(); var_dump($foo::class);
DateTime 객체 생성을 위한 인터페이스
이미 DateTime::createFromImmutable($immutableDateTime)을 사용하여 DateTimeImmutable 객체에서 DateTime 객체를 생성할 수 있지만 그 반대는 까다롭습니다.
DateTime::createFromInterface() 및 DatetimeImmutable::createFromInterface()가 추가되면서 이제 DateTime 및 DateTimeImmutable 개체를 서로 변환하는 보편적인 방법이 있습니다.
DateTime::createFromInterface(DateTimeInterface $other); DateTimeImmutable::createFromInterface(DateTimeInterface $other);
새로운 문자열 가능 인터페이스 RFC
문자열 가능 인터페이스는 임의의 문자열을 입력하거나 __toString()을 구현하는 데 사용할 수 있습니다.
또한 클래스가 __toString()을 구현할 때마다 백그라운드에서 자동으로 인터페이스를 구현하므로 수동으로 구현할 필요가 없습니다.
class Foo { public function __toString(): string { return 'foo'; } } function bar(Stringable $stringable) { /* … */ } bar(new Foo()); bar('abc');
새로운 str_contains() 함수 RFC
어떤 사람들은 이것이 너무 오래되었다고 말할 수도 있지만, 마침내 우리는 한 문자열에 다른 문자열이 포함되어 있는지 알기 위해 더 이상 strpos()에 의존할 필요가 없습니다.
이전:
if (strpos('string with lots of words', 'words') !== false) { /* … */ }
현재:
if (str_contains('string with lots of words', 'words')) { /* … */ }
새 fdiv() 함수 PR
새 fdiv() 함수는 fmod() 및 intdiv() 함수처럼 작동하며 0으로 나눌 수 있습니다.
경우에 따라 오류 대신 INF, -INF 또는 NaN이 표시됩니다.
새로운 get_debug_type() 함수 RFC
get_debug_type()은 변수 유형을 반환합니다.
gettype()이 할 수 있는 것처럼 들리네요.
get_debug_type()은 배열, 문자열, 익명 클래스 및 객체에 대해 더 유용한 출력을 반환합니다.
예를 들어, fooBar 클래스에서 gettype()을 호출하면 Object가 반환됩니다.
get_debug_type()을 사용하면 클래스 이름이 반환됩니다.
get_debug_type()과 gettype()의 전체 차이점 목록은 RFC에서 찾을 수 있습니다.
특성의 추상 메서드 개선 RFC
특성은 이를 사용하는 클래스에서 구현해야 하는 추상 메서드를 지정할 수 있습니다.
하지만 주의할 점이 있습니다. PHP8 이전에는 이러한 메소드 구현의 서명이 확인되지 않았습니다.
在以下代码中有效:
trait Test { abstract public function test(int $input): int; } class UsesTrait { use Test; public function test($input) { return $input; } }
在使用 traits 并实现其抽象方法时,PHP8 将执行正确的方法签名验证。
这意味着您需要改写以下内容:
class UsesTrait { use Test; public function test(int $input): int { return $input; } }
token_get_all () 的对象接口 RFC
函数的作用是:返回值的是一个数组。
此 RFC 使用 PhpToken::getall () 方法添加一个 PhpToken 类。
此实现使用对象,而不是普通值。
它消耗更少的内存,更容易阅读。
变量语法调整 RFC
来自 RFC:“统一变量语法 RFC 解决了 PHP 变量语法中的一些不一致问题”,这个 RFC 打算解决少数被忽略的情况。
内部函数的类型批注
很多人都参与到为所有内部函数添加适当类型注释的工作中。
这是一个长期存在的问题,通过在以前版本中对 PHP 所做的所有更改,最终可以解决这个问题。
这意味着内部函数和方法在反射中将具有完整的类型信息。
统一错误类型 RFC
PHP 中的用户定义函数已经抛出 TypeErrors,但是内部函数没有抛出 TypeErrors,而是发出警告并返回 NULL。
从 PHP8 开始,内部函数的行为已经保持一致。
重新分类 zend engine 报错 RFC
许多以前只触发警告或通知的错误已转换为适当的错误。
以下警告已更改。
未定义变量:错误异常而不是通知。
未定义的数组索引:警告而不是通知。
被零除:DivisionByZeroError 异常而不是警告。
尝试递增 / 递减非对象的属性‘% s’:错误异常而不是警告。
试图修改非对象的属性‘% s’:错误异常而不是警告。
尝试分配非对象的属性‘% s’:错误异常而不是警告。
从空值创建默认对象:错误异常而不是警告。
正在尝试获取非对象的属性‘% s’:警告而不是通知。
未定义属性:% s::$% s:警告而不是通知。
无法将元素添加到数组,因为下一个元素已被占用:错误异常而不是警告。
无法取消设置非数组变量中的偏移量:错误异常而不是警告。
不能将标量值用作数组:错误异常而不是警告。
只能解包数组和遍历:TypeError 异常而不是警告。
为 foreach () 提供的参数无效:TypeError 异常而不是警告。
偏移类型非法:TypeError 异常而不是警告。
isset 中的偏移类型非法或为空:TypeError 异常而不是警告。
未设置中的偏移类型非法:TypeError 异常而不是警告。
数组到字符串的转换:警告而不是通知。
资源 ID#% d 用作偏移量,转换为整数 (% d):警告而不是通知。
发生字符串偏移量转换:警告而不是通知。
未初始化的字符串偏移量:% d:警告而不是通知。
无法将空字符串分配给字符串偏移量:错误异常而不是警告
默认错误报告级别
现在是 E_ALL,而不是除 E_NOTICE 和 E_DEVERATED 之外的所有内容。
这意味着可能会弹出许多以前被悄悄忽略的错误,尽管在 PHP8 之前可能已经存在。
@运算符不再忽略致命错误
此更改可能会揭示在 PHP8 之前隐藏的错误。请确保在生产服务器上设置 display_errors=off !
串联优先级 RFC
虽然在 PHP7.4 中已不推荐使用,但此更改现在生效。
如果你这样写的话:
echo "sum: " . $a + $b;
PHP 以前会这样解释它:
echo ("sum: " . $a) + $b;
PHP 8 将会这样解释它:
echo "sum: " . ($a + $b);
反射方法签名更改
反射类的三个方法签名已更改:
ReflectionClass::newInstance($args); ReflectionFunction::invoke($args); ReflectionMethod::invoke($object, $args);
现已成为:
ReflectionClass::newInstance(...$args); ReflectionFunction::invoke(...$args); ReflectionMethod::invoke($object, ...$args);
升级指南指定,如果您扩展了这些类,并且仍然希望同时支持 PHP 7 和 PHP 8,则允许以下签名:
ReflectionClass::newInstance($arg = null, ...$args); ReflectionFunction::invoke($arg = null, ...$args); ReflectionMethod::invoke($object, $arg = null, ...$args);
推荐教程:《PHP教程》
위 내용은 PHP 8이 반년만에 출시됩니다. 어떤 새로운 기능이 있는지 살펴보겠습니다.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!