
php 8.1 引入的 `readonly` 关键字,旨在简化不可变对象的创建。它允许属性在初始化后保持不变,有效防止意外修改,减少传统 getter 方法的样板代码,并提升代码的清晰度和安全性。php 8.2 进一步引入了 `readonly` 类,使得整个类的公共属性默认为只读,为构建更健壮的应用提供了强大支持。
在软件开发中,不可变性(Immutability)是一个重要的概念,它指的是对象在创建后,其内部状态不能被修改。不可变对象带来了诸多好处,例如提高并发安全性、简化状态管理、减少错误等。然而,在 PHP 中实现不可变属性往往需要编写额外的样板代码。
为了解决这一痛点,PHP 8.1 引入了 readonly 关键字,允许开发者在属性声明时明确指定其为只读。这意味着一旦属性被初始化(通常在构造函数中),其值便不能再被修改。这一特性显著提升了代码的简洁性和安全性,为构建不可变对象提供了原生支持。
readonly 属性的主要价值体现在以下几个方面:
readonly 属性与 const(常量)都用于定义不可变的值,但它们之间存在重要的区别:
立即学习“PHP免费学习笔记(深入)”;
为了更好地理解 readonly 带来的便利,我们来看一下实现不可变属性的演进过程。
在 PHP 8.1 之前,如果我们需要一个在创建后不能被修改的属性,通常会将其声明为私有,并提供一个公共的只读访问器(getter):
class Foo
{
private DateTimeImmutable $createdAt;
public function __construct()
{
$this->createdAt = new DateTimeImmutable();
}
public function getCreatedAt(): DateTimeImmutable
{
return $this->createdAt;
}
}
$f = new Foo();
echo $f->getCreatedAt()->format('Y-m-d H:i:s');
// 尝试修改会报错或无效
// $f->createdAt = new DateTimeImmutable(); // 错误:不能访问私有属性这种方式虽然实现了不可变性,但需要为每个不可变属性编写额外的私有属性声明和公共 getter 方法,增加了不少样板代码。
PHP 8.1 引入 readonly 关键字后,结合构造函数属性提升,可以大大简化上述代码:
class Foo
{
public function __construct(
public readonly DateTimeImmutable $createdAt = new DateTimeImmutable()
) { }
}
$f = new Foo();
echo $f->createdAt->format('Y-m-d H:i:s');
// 尝试修改会抛出错误:
// Fatal error: Uncaught Error: Cannot modify readonly property Foo::$createdAt
// $f->createdAt = new DateTimeImmutable();通过 public readonly 声明,createdAt 属性在构造函数中初始化后便不可修改。代码变得更加简洁,意图也更加明确。
PHP 8.2 更进一步,引入了 readonly 类。当一个类被声明为 readonly 时,该类中的所有公共(public)属性都将自动变为只读,无需为每个属性单独添加 readonly 关键字。
readonly class Foo
{
public function __construct(
public string $name,
public DateTimeImmutable $createdAt = new DateTimeImmutable()
) { }
}
$f = new Foo('My Foo Instance');
echo $f->name . ' created at ' . $f->createdAt->format('Y-m-d H:i:s');
// 尝试修改 $f->name 会抛出错误:
// Fatal error: Uncaught Error: Cannot modify readonly property Foo::$name
// $f->name = 'New Name';
// 尝试修改 $f->createdAt 也会抛出错误:
// Fatal error: Uncaught Error: Cannot modify readonly property Foo::$createdAt
// $f->createdAt = new DateTimeImmutable();readonly 类使得创建完全不可变的对象变得异常简单,尤其适用于值对象(Value Objects)或数据传输对象(DTOs)等场景。
在使用 readonly 属性和类时,需要注意以下几点:
PHP 8.1 引入的 readonly 属性以及 PHP 8.2 扩展的 readonly 类,是 PHP 语言在支持不可变性方面迈出的重要一步。它们提供了一种简洁、安全且高效的方式来创建不可变对象,有效减少了样板代码,提升了代码的清晰度和可维护性。通过合理利用 readonly 特性,开发者可以构建出更健壮、更易于理解和测试的 PHP 应用程序。
以上就是PHP 8.1 readonly 属性详解:构建不可变对象的现代实践的详细内容,更多请关注php中文网其它相关文章!
PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号