>헤드라인 >모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

藏色散人
藏色散人앞으로
2020-10-30 14:39:488067검색

PHP 8의 모든 새로운 기능 및 코드 예제 개요

PHP 8의 공식 버전이 곧 출시될 예정입니다. RC3은 10월 29일에 출시될 예정이며, RC4는 11월 12일에 출시될 예정이며, 공식 버전도 출시될 예정입니다. 11월 26일(PHP 8 출시 일정클릭하여 확인 가능)

이제 PHP 8에 추가될 새로운 기능을 살펴보겠습니다. 이 기사에서는 PHP 8의 모든 새로운 기능과 해당 코드 예제를 보여줍니다.

PHP 8 컴파일 및 설치

샘플 코드를 편리하게 실행하기 위해 시작하기 전에 PHP 8 RC2 버전을 로컬에서 컴파일하고 설치할 수 있습니다.

// 0、下载解压源码
wget https://downloads.php.net/~pollita/php-8.0.0RC2.tar.gz
tar zxvf php-8.0.0RC2.tar.gz
cd php-8.0.0RC2

// 1、生成 configure 文件
./buildconf --force

// 2、配置构建流程(最小化安装)
./configure --prefix=/usr/local/php8 \
--with-config-file-path=/usr/local/php8 \
--enable-cli \
--without-iconv

// 3、构建 && 安装
make && sudo make install

// 4、拷贝配置文件
sudo cp php.ini-development /usr/local/php8/php.ini

설치가 완료된 후 PHP 8이 설치되어 있는지 확인하세요. 버전 확인을 통해 성공적으로 설치되었습니다.

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

로컬 운영 체제는 Mac입니다. Windows 환경 설치의 경우 이전에 JIT를 소개한 이 튜토리얼을 참조하세요. WSL 가상 머신 기반 PHP 8 컴파일 및 설치.

PHP 8 CLI 인터프리터 호출을 용이하게 하기 위해 ZSH 구성 파일 ~/.zshrc에 별칭을 구성했습니다. ~/.zshrc 中为其配置了别名:

alias php8="/usr/local/php8/bin/php"

然后运行 source ~/.zshrc 让上述别名生效,以后就可以直接通过 php8 调用 PHP 8 CLI 解释器了:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

新建示例项目

接下来,我们在 PhpStorm 中新建一个 php8-demo 项目来存放本篇教程示例代码,并且将 PHP 语言级别和命令行解释器都调整为 PHP 8.0,以便让 PhpStorm 支持最新版本 PHP(不知道怎么配置的?可以到PhpStorm教程里面去翻一翻哈):

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

然后就可以开始我们的 PHP 8 新特性探索之旅了。

新增对联合类型的支持

联合类型允许一个变量拥有多个类型的值,而不是一个(参考 C 语言的联合类型很好理解)。

我们编写一段示例代码如下:

<?php declare(strict_types=1);

/**
 * 定义一个支持联合类型的 Number 类
 */ 
class Number {
    private int|float $number;
 
    public function setNumber(int|float $number): void {
        $this->number = $number;
    }
 
    public function getNumber(): int|float {
        return $this->number;
    }
}

/**
 * 我们可以传递浮点型和整型值到 Number 对象
 */
$number = new Number();

$number->setNumber(5);

var_dump($number->getNumber());

$number->setNumber(11.54);

var_dump($number->getNumber());

exit;

上述代码运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

新增 WeakMap 特性

WeakMap 允许你创建对象到任意值的映射(类似 SplObjectStorage),同时也不会阻止作为键的对象被垃圾回收。如果某个对象键被垃圾回收,对应键值对将从集合中移除。

这一新特性非常有用,因为这样一来,开发者就不必担心代码存在内存泄露了。大多数 PHP 开发者可能对此并不关心,但是当你编写长时间运行的进程时一定要提防这个问题,比如使用 ReactPHP 进行事件驱动编程时:有了 WeakMap 后,引用的对象会在失效时自动被垃圾回收。

如果你在数组中做同样的事情,则仍然会持有该对象的引用,从而导致内存泄露。

我们编写一段示例代码如下:

<?php declare(strict_types=1);


class FooBar {
    public WeakMap $cache;

    public function __construct() {
        $this->cache = new WeakMap();
    }

    public function getSomethingWithCaching(object $obj) {
        return $this->cache[$obj] ??= $this->computeSomethingExpensive($obj);
    }

    public function computeSomethingExpensive(object $obj) {
        var_dump("I got called");
        return rand(1, 100);
    }
}

$cacheObject = new stdClass;

$obj = new FooBar;
// "I got called" 只会打印一次
$obj->getSomethingWithCaching($cacheObject);
$obj->getSomethingWithCaching($cacheObject);

var_dump(count($obj->cache));

// 删除该对象后 WeakMap 会释放相应内存
unset($cacheObject);

var_dump(count($obj->cache));

exit;

对应的运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

新增 ValueError 异常

PHP 8 引入了新的名为 ValueError 的内置异常类,它继承自 Exception 基类。每次当你传递值到函数时,如果是一个无效类型,则会抛出该异常,在 PHP 8 之前,这样的操作会导致警告。

下面是示例代码:

<?php declare(strict_types=1);

/**
 * 传递数组到 array_rand,类型正确,但是 array_rand 期望传入的是非空数组
 * 所以会抛出 ValueError 异常
 */
array_rand([], 0);

/**
 * json_decode 的深度参数必须是有效的正整型值,
 * 所以这里也会抛出 ValueError 异常
 */
json_decode(&#39;{}&#39;, true, -1);

运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

重写方法时允许可变参数

当我们在子类重写父类方法时,任何数量的参数现在都可以被替换成可变参数,只要对应参数类型是兼容的即可:

<?php declare(strict_types=1);

class A {
    public function method(int $many, string $parameters, $here) {

    }
}
class B extends A {
    public function method(...$everything) {
        var_dump($everything);
    }
}

$b = new B();
$b->method('i can be overwritten!');
exit;

运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

静态返回类型

PHP 8 中可以使用 static 关键字标识某个方法返回该方法当前所属的类,即使它是继承的(后期静态绑定):

<?php declare(strict_types=1);

class Test {
    public function doWhatever(): static {
        // Do whatever.
        return $this;
    }
}

exit;

对象的类名字面量

PHP 8 中可以使用 $object::class 获取对象的类名,其返回结果和 get_class($object)

<?php declare(strict_types=1);

class Test {

}

$test = new Test();

var_dump($test::class);
var_dump(get_class($test));
exit;
그런 다음 source ~/.zshrc를 실행합니다. code> 위의 별칭을 적용하려면 앞으로 php8를 통해 PHP 8 CLI 인터프리터를 직접 호출할 수 있습니다.

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

새 샘플 프로젝트

다음으로 새로운 php8-demo를 생성합니다. 저장용 PhpStorm의 code> 프로젝트 이 튜토리얼에서는 샘플 코드를 제공하고, PhpStorm이 최신 버전의 PHP를 지원하도록 PHP 언어 수준과 명령줄 인터프리터를 PHP 8.0으로 조정합니다. 구성 방법을 모르시나요? PhpStorm

튜토리얼 살펴보기):🎜🎜모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요🎜🎜그런 다음 PHP 8의 새로운 기능을 탐색하는 여정을 시작할 수 있습니다. 🎜🎜🎜Union 유형에 대한 새로운 지원🎜🎜🎜Union 유형을 사용하면 변수가 하나가 아닌 여러 유형의 값을 가질 수 있습니다(잘 이해하려면 C 언어의 Union 유형을 참조하세요). 🎜🎜다음과 같이 샘플 코드를 작성합니다. 🎜
<?php declare(strict_types=1);

class Foo {}
class Bar {}

$names = [&#39;Foo&#39;, &#39;Bar&#39;];
$class = new ($names[array_rand($names)]);

var_dump($class);

exit;
🎜위 코드는 다음과 같이 실행됩니다. 🎜🎜모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요🎜🎜🎜새로운 WeakMap 기능🎜🎜🎜WeakMap을 사용하면 객체를 다음과 같이 차단하지 않고 임의의 값(SplObjectStorage와 유사)에 대한 객체 매핑을 생성할 수 있습니다. 키 가비지 수집이 완료되었습니다. 객체 키가 가비지 수집되면 해당 키-값 쌍이 컬렉션에서 제거됩니다. 🎜🎜이 새로운 기능은 개발자가 코드의 메모리 누수에 대해 걱정할 필요가 없기 때문에 매우 유용합니다. 대부분의 PHP 개발자는 아마도 이 문제에 신경 쓰지 않을 것입니다. 그러나 이벤트 중심 프로그래밍을 위해 ReactPHP를 사용할 때와 같이 장기 실행 프로세스를 작성할 때 이 문제에 주의해야 합니다. WeakMap을 사용하면 참조된 개체가 만료될 때 자동으로 만료됩니다. .가비지 수집이 완료되었습니다. 🎜🎜배열에서 동일한 작업을 수행하면 객체에 대한 참조가 계속 유지되어 메모리 누수가 발생합니다. 🎜🎜다음과 같이 샘플 코드를 작성합니다. 🎜
<?php declare(strict_types=1);

class Foo {
    public function __toString() {
        return &#39;I am a class&#39;;
    }
}

$obj = new Foo;
var_dump($obj instanceof Stringable);

exit;
🎜해당 실행 결과는 다음과 같습니다. 🎜🎜모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요🎜🎜🎜새로운 ValueError 예외🎜🎜🎜PHP 8에는 ValueError라는 이름의 새로운 내장 예외 클래스가 도입되었습니다. 이 클래스는 Exception에서 상속됩니다. 코드 > 기본 클래스. 이 예외는 잘못된 유형의 함수에 값을 전달할 때마다 발생합니다. PHP 8 이전에는 이러한 작업으로 인해 경고가 발생했습니다. 🎜🎜다음은 샘플 코드입니다. 🎜<pre class="brush:php;toolbar:false">&lt;?php declare(strict_types=1); trait MyTrait { abstract private function neededByTheTrait(): string; public function doSomething() { return strlen($this-&gt;neededByTheTrait());     } } class TraitUser {     use MyTrait;     // 支持该语法     private function neededByTheTrait(): string { }     // 不支持该语法 (错误的返回类型)     // private function neededByTheTrait(): stdClass { }     // 支持该语法 (非静态方法变成了静态方法)     // private static function neededByTheTrait(): string { } } exit;</pre>🎜실행 결과는 다음과 같습니다. 🎜🎜<img src="https://img.php.cn/upload/article/000/000/020/10c92ed684f00b74ebec12ad04347933-5.%20jpg" alt=" 모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요">🎜🎜🎜메서드를 재정의할 때 변수 매개 변수 허용🎜🎜🎜하위 클래스에서 상위 클래스 메서드를 재정의할 때 이제 해당하는 한 모든 매개 변수를 변수 매개 변수로 바꿀 수 있습니다. 매개변수 유형은 호환 가능합니다. 🎜<pre class="brush:php;toolbar:false">&lt;?php declare(strict_types=1); $callable = fn() =&gt; throw new Exception(); $nullableValue = null; // $value 是非空的 $value = $nullableValue ?? throw new \InvalidArgumentException(); exit;</pre>🎜실행 결과는 다음과 같습니다. 🎜🎜<img src="https://img.php.cn/upload/article/000/000/020/10c92ed684f00b74ebec12ad04347933-6.jpg" alt="모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요 ">🎜🎜🎜정적 반환 유형🎜🎜🎜PHP 8에서는 <code>static 키워드를 사용하여 해당 메서드가 현재 속한 클래스를 반환하는 메서드를 식별할 수 있습니다. 상속된 경우(후기 정적 바인딩) :🎜
<?php declare(strict_types=1);

function method_with_many_arguments(
    $a,
    $b,
    $c,
    $d,
) {
    var_dump("this is valid syntax");
}

method_with_many_arguments(
    1,
    2,
    3,
    4,
);

exit;
🎜🎜객체의 클래스 이름 리터럴🎜🎜🎜PHP 8에서는 $object::class를 사용하여 객체의 클래스 이름을 얻을 수 있으며, 반환 결과는 get_class($object)와 동일합니다. 동일: 🎜<pre class="brush:php;toolbar:false">&lt;?php declare(strict_types=1); $nullableValue = null; try { $value = $nullableValue ?? throw new \InvalidArgumentException(); } catch (\InvalidArgumentException) { var_dump(&quot;Something went wrong&quot;); } exit;</pre>🎜실행 결과는 다음과 같습니다. 🎜🎜🎜🎜🎜🎜변수 구문 조정🎜🎜<p><code>newinstanceof 关键字现在可以被用于任意表达式:

<?php declare(strict_types=1);

class Foo {}
class Bar {}

$names = [&#39;Foo&#39;, &#39;Bar&#39;];
$class = new ($names[array_rand($names)]);

var_dump($class);

exit;

运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

Stringable 接口

PHP 8 引入了新的 Stringable 接口,只要某个类实现了 __toString 方法,即被视作自动实现了 Stringable 接口(咋和 Go 接口实现有点像),而不必显式声明实现该接口:

<?php declare(strict_types=1);

class Foo {
    public function __toString() {
        return &#39;I am a class&#39;;
    }
}

$obj = new Foo;
var_dump($obj instanceof Stringable);

exit;

运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

Trait 现在可以定义抽象私有方法

<?php declare(strict_types=1);


trait MyTrait {
    abstract private function neededByTheTrait(): string;

    public function doSomething() {
        return strlen($this->neededByTheTrait());
    }
}

class TraitUser {
    use MyTrait;

    // 支持该语法
    private function neededByTheTrait(): string { }

    // 不支持该语法 (错误的返回类型)
    // private function neededByTheTrait(): stdClass { }

    // 支持该语法 (非静态方法变成了静态方法)
    // private static function neededByTheTrait(): string { }
}

exit;

throw 现在可以被用作表达式

throw 语句现在可以用在只允许表达式出现的地方,例如箭头函数、合并运算符和三元运算符等:

<?php declare(strict_types=1);

$callable = fn() => throw new Exception();

$nullableValue = null;

// $value 是非空的
$value = $nullableValue ?? throw new \InvalidArgumentException();


exit;

参数列表中允许出现可选的尾部逗号

和数组中的尾部逗号类似,现在也可以在参数列表中定义一个尾部逗号:

<?php declare(strict_types=1);

function method_with_many_arguments(
    $a,
    $b,
    $c,
    $d,
) {
    var_dump("this is valid syntax");
}

method_with_many_arguments(
    1,
    2,
    3,
    4,
);

exit;

上述代码运行结果是正常的:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

捕获异常而不存储到变量

现在可以编写 catch (Exception) 代码来捕获异常而不必将其存储到一个变量中:

<?php declare(strict_types=1);

$nullableValue = null;

try {
    $value = $nullableValue ?? throw new \InvalidArgumentException();
} catch (\InvalidArgumentException) {
    var_dump("Something went wrong");
}

exit;

上述代码运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

新增对 mixed 类型的支持

PHP 8 引入了新的名为 mixed 的类型,该类型等价于 array|bool|callable|int|float|null|object|resource|string

<?php declare(strict_types=1);

function debug_function(mixed ...$data)
{
    var_dump($data);
}

debug_function(1, &#39;string&#39;, []);

exit;

上述代码运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

新增对注解的支持

PHP 8 的注解实际上包含了多个 RFC:

  • https://wiki.php.net/rfc/attributes_v2
  • https://wiki.php.net/rfc/attribute_amendments
  • https://wiki.php.net/rfc/shorter_attribute_syntax
  • https://wiki.php.net/rfc/shorter_attribute_syntax_change

注解绝对是 PHP 8 引入的最大新特性之一,一开始理解起来可能有点困难(不过有 Java 基础的话会很简单)。简而言之,注解允许你添加元数据到 PHP 函数、参数、类等,这些元数据随后可以通过可编程方式获取,在 PHP 7 或者更低版本中实现类似功能需要解析代码注释块,而通过注解可以直接访问深度集成到 PHP 自身的这些信息。

我们来编写一段示例代码方便你理解,假设你想要允许开发者添加中间件到控制器类/方法,使用注解可以这么做:

<?php declare(strict_types=1);

// 首先,我们需要定义注解,注解本身只是一个原生的 PHP 类,并且自身被打上了注解的注释

#[Attribute]
class ApplyMiddleware
{
    public array $middlware = [];

    public function __construct(...$middleware)
    {
        $this->middleware = $middleware;
    }
}

// 下面的语法会添加上述注解到 MyController 类,并且传入 auth 作为参数

#[ApplyMiddleware('auth')]
class MyController
{
    public function index()
    {
    }
}

// 然后我们就可以在类中使用反射获取所有的 ApplyMiddleware 注解并读取给定的中间件参数

$reflectionClass = new ReflectionClass(MyController::class);

$attributes = $reflectionClass->getAttributes(ApplyMiddleware::class);

foreach ($attributes as $attribute) {
    $middlewareAttribute = $attribute->newInstance();
    var_dump($middlewareAttribute->middleware);
}

exit;

运行上述代码,打印结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

新增构造函数属性提示支持

这个新特性只是一个语法简写而言,可以将属性声明和构造函数属性初始化合并到一起:

<?php declare(strict_types=1);

class User {
    public function __construct(
        public int $id,
        public string $name,
    ) {}
}

$user = new User(1, &#39;Marcel&#39;);

var_dump($user->id);
var_dump($user->name);

exit;

上述代码运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

新增 match 表达式支持

match 表达式和 switch 分支语句类型,但是语义上更加安全并且可以直接返回值:

<?php declare(strict_types=1);

echo match (1) {
    0 => 'Foo',
    1 => 'Bar',
    2 => 'Baz',
};

exit;

上述代码运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

新增对空安全运算符 ?-> 的支持

当该运算符的左侧评估为 null 时,整个代码链路的执行将会被终止并整体评估为 null。如果不为 null 的话,则和普通的 -> 运算符功能一致:

<?php declare(strict_types=1);

class User {
    public function getAddress() {}
}

$user = new User();

$country = $user?->getAddress()?->country?->iso_code;

var_dump($country);

exit;

上述代码运行结果如下:

모든 새로운 PHP 8 기능 및 코드 예제에 대한 개요

新增对命名参数的支持

命名参数允许基于参数名称传递参数到函数,而不是参数所在的位置,这样一来,函数参数就可以自解释并且与顺序无关,并且允许跳过默认值:

<?php declare(strict_types=1);

array_fill(start_index: 0, num: 100, value: 50);

exit;

注:PHP 8 还有另一个重要的新特性 —— 引入对 JIT 的支持,不过对于上层业务代码而言是无感的,只是底层优化而已,想要了解 JIT 对 PHP 应用性能的影响,请参考之前发布的这篇文章:https://xueyuanjun.com/post/21702。本篇教程所有示例代码整理自 PHP 8 - try out all new features,原文基于 Laravel 框架进行测试,这里将其转化为了通用的 PHP 代码


성명:
이 기사는 xueyuanjun.com에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제