PHP8.1新的initializers初始化器,怎么使用 ?

198 阅读2分钟

开启掘金成长之旅!这是我参与「掘金日新计划 · 2 月更文挑战」的第 13 天,点击查看活动详情
PHP 8.1 添加了一个看似很小的细节,但我认为它会对许多人产生重大的日常影响。那么这个“初始化器 RFC 中的新内容”是关于什么的?我们来看一个例子;我们都写过这样的代码:

class MyStateMachine

{

    public function __construct(

        private ?State $state = null,

    ) {

        $this->state ??= new InitialState();

    }

}

在这个状态机示例中,我们想以两种方式构造我们的类:有和没有初始状态。如果我们在没有初始状态的情况下构建它,我们希望设置一个默认状态。

PHP 当然支持直接在参数列表中设置初始值,但仅适用于原始类型。例如,如果我们的状态机在内部使用字符串而不是对象,我们就可以像这样编写它的构造函数:

class MyStateMachine{

    public function __construct(

        private string $state 'initial',

    ) {

    }

}

因此,在 PHP 8.1 中,我们也可以对对象使用相同的“默认值”语法。换句话说:您可以使用new默认参数(这是“初始化程序”的一个示例):

class MyStateMachine{

    public function __construct(

        private State $state = new InitialState(),

    ) {

    }

}

“初始化器”不仅仅是参数默认值,这里有一个来自 RFC 的简单解释:该 RFC 提议允许在参数默认值、属性参数、静态变量初始值设定项和全局常量初始值设定项中使用新表达式你没看错:属性也在这个列表中!想象一个简单的验证库,它使用特性来验证属性的输入。也许它应该能够验证数组元素,如下所示:

class CreateEmailsRequest extends FormRequestData

{

    #[ValidArray(

        email: [new Required, new ValidEmail],

        name: [new Required, new ValidString],

    )]

    public array $people;

}

在 PHP 8.1 之前,您将无法编写此类代码,因为new由于它们的评估方式,您不允许在属性中使用,但现在您可以了!让我们来看看一些值得一提的重要细节。您想要了解更多有关 PHP 8.1 的信息吗?有通往 PHP 8.1 的道路。在接下来的 10 天内,您将每天收到一封电子邮件,内容涉及 PHP 8.1 的一个新的和现有的特性;之后您将自动退订,因此不会收到垃圾邮件或后续邮件。 现在订阅!

仅在需要时构建

这些“新价值观”只会在实际需要时才会被构建。这意味着,在我们的第一个示例中,InitialState如果没有给出参数,PHP 只会创建一个新对象:

class MyStateMachine

{

    public function __construct(

        private State $state = new InitialState(),

    ) {

    }

}

new MyStateMachine(new DraftState()); // No InitialState is created

new MyStateMachine(); // But now it is

例如,在属性的情况下,只有newInstance在反射属性上调用时才会创建对象。

不在类属性中

您还应该知道不能new在类属性中用作默认值。支持此功能会引入许多无法预料的副作用,例如,在序列化和反序列化对象时。

class MyStateMachine{
    private State $state = new InitialState();
}

幸运的是,我们提升了允许默认值的属性,因为 PHP 将转换属性提升语法,在构造函数参数中保留默认值,但不在实际属性中。以下是转译后的版本:

class MyStateMachine{

    private State $state;

    public function __construct(

        State $state new InitialState(),

    ) {

        $this->state = $state;

    }

}

有限的投入

您可能已经猜到了,但是在初始化程序中构造新对象时,您只能传递一组有限的输入。例如,您不能使用变量、展开运算符、匿名类等。不过,这是一个非常受欢迎的补充!