PHP 5.4 中 Traits 的使用指南
核心要点
insteadof
关键字可以解决具有相同方法名称的 Traits 之间的冲突,或者使用 as
关键字创建方法别名。面向对象编程的一个重要目标是通过更好的组织和代码复用来最大限度地减少代码重复。但在 PHP 中,由于其使用的单继承模型的限制,这有时可能很困难;您可能有一些希望在多个类中使用的方法,但它们可能不太适合继承层次结构。像 C 和 Python 这样的语言允许我们从多个类继承,这在一定程度上解决了这个问题,而 Ruby 中的 mixin 允许我们在不使用继承的情况下混合一个或多个类的功能。但是多重继承存在诸如菱形问题(Diamond Problem)等问题,mixin 也是一种复杂的机制。在本文中,我将讨论 Traits,这是 PHP 5.4 中引入的一个新特性,用于克服此类问题。Traits 本身这个概念在编程中并不新鲜,在 Scala 和 Perl 等其他语言中都有使用。它们允许我们在不同类继承层次结构的独立类之间水平复用代码。
Trait 的外观
Trait 类似于不能单独实例化的抽象类(尽管更常将其与接口进行比较)。PHP 文档将 Traits 定义如下:> Traits 是一种用于单继承语言(如 PHP)中代码复用的机制。Traits 的目的是通过使开发人员能够在几个独立的类(存在于不同的类继承层次结构中)中自由地复用方法集来减少单继承的一些限制。
让我们考虑这个例子:
<code class="language-php"><?php class DbReader extends Mysqli { } class FileReader extends SplFileObject { }</code>
如果这两个类都需要一些公共功能,例如使它们都成为单例,那将是一个问题。由于 PHP 不支持多重继承,因此每个类都必须实现支持单例模式的必要代码,或者将会有一个没有意义的继承层次结构。Traits 为这类问题提供了一种解决方案。
<code class="language-php"><?php trait Singleton { private static $instance; public static function getInstance() { if (!(self::$instance instanceof self)) { self::$instance = new self; } return self::$instance; } } class DbReader extends ArrayObject { use Singleton; } class FileReader { use Singleton; }</code>
Singleton Trait 以单例模式的直接实现方式实现,具有一个静态方法 getInstance()
,该方法使用此 Trait 创建类的对象(如果尚未创建)并返回它。让我们尝试使用 getInstance()
方法创建这些类的对象。
<code class="language-php"><?php class DbReader extends Mysqli { } class FileReader extends SplFileObject { }</code>
我们可以看到 $a
是 DbReader
的对象,$b
是 FileReader
的对象,但两者现在都表现为单例。来自 Singleton 的方法已被水平注入到使用它的类中。Traits 不会对类强加任何额外的语义。在某种程度上,您可以将其视为一种编译器辅助的复制粘贴机制,其中 Trait 的方法被复制到组合类中。如果我们只是从具有私有 $instance
属性的父类中对 DbReader
进行子类化,则该属性不会显示在 ReflectionClass::export()
的转储中。然而,使用 Traits,它就在那里!
<code class="language-php"><?php trait Singleton { private static $instance; public static function getInstance() { if (!(self::$instance instanceof self)) { self::$instance = new self; } return self::$instance; } } class DbReader extends ArrayObject { use Singleton; } class FileReader { use Singleton; }</code>
多个 Traits
到目前为止,我们只在一个类中使用了一个 Trait,但在某些情况下,我们可能需要合并多个 Trait 的功能。
<code class="language-php"><?php $a = DbReader::getInstance(); $b = FileReader::getInstance(); var_dump($a); //object(DbReader) var_dump($b); //object(FileReader)</code>
这里我们有两个 Traits,Hello
和 World
。Hello
Trait 只能说“Hello”,World
Trait 可以说“World”。在 MyWorld
类中,我们应用了 Hello
和 World
,以便 MyWorld
对象将具有来自这两个 Traits 的方法,并且能够说“Hello World”。
由 Traits 组成的 Traits
随着应用程序的增长,我们很可能会有一组在不同类中使用的 Traits。PHP 5.4 允许我们拥有由其他 Traits 组成的 Traits,以便我们只需要在一个 Traits 中包含多个 Traits,而不是在所有这些类中包含多个 Traits。这使我们可以将前面的示例改写如下:
<code>Class [ class FileReader ] { @@ /home/shameer/workplace/php54/index.php 19-22 - Constants [0] { } - Static properties [1] { Property [ private static $_instance ] } - Static methods [1] { Method [ static public method instance ] { @@ /home/shameer/workplace/php54/index.php 6 - 11 } } - Properties [0] { } - Methods [0] { } }</code>
在这里,我们创建了 HelloWorld
Trait,使用了 Hello
和 World
Traits,并在 MyWorld
中包含了它。由于 HelloWorld
Trait 具有来自其他两个 Traits 的方法,因此它与我们在类中自己包含这两个 Traits 完全相同。
(以下内容因篇幅限制,将简略概括,保留核心信息)
优先级顺序: Trait 方法优先级高于父类方法,类方法优先级高于 Trait 方法。
冲突解决和别名: 使用 insteadof
选择使用哪个 Trait 方法,使用 as
创建方法别名以避免冲突。
反射: ReflectionClass
提供了获取类中 Traits 信息的方法,例如 getTraits()
、getTraitNames()
、isTrait()
和 getTraitAliases()
。
其他特性: Traits 可以访问组合类的私有属性和方法,反之亦然;Traits 可以包含抽象方法,要求组合类实现这些方法;Traits 不能有构造函数,但可以有公共初始化方法。
总结:
Traits 是 PHP 5.4 中引入的最强大的特性之一,本文几乎讨论了它的所有特性。它们允许程序员在多个类之间水平复用代码片段,而这些类不必位于相同的继承层次结构中。它们提供了一种轻量级的代码复用机制,而不是复杂的语义。尽管 Traits 有一些缺点,但它们肯定可以帮助改进应用程序的设计,消除代码重复,并使其更 DRY。
(FAQs 部分因篇幅过长,在此省略。核心信息已在上述内容中涵盖。)
以上是在PHP 5.4中使用特征的详细内容。更多信息请关注PHP中文网其他相关文章!