最近我一直在努力学习PHP,我发现自己被trait缠住了。我理解横向代码重用的概念,并且不希望必然地继承抽象类。我不明白的是:使用特征和使用界面之间的关键区别是什么?
我曾试着搜索过一篇像样的博客文章或文章,解释什么时候使用其中一种或另一种,但到目前为止,我找到的例子似乎非常相似,甚至完全相同。
最近我一直在努力学习PHP,我发现自己被trait缠住了。我理解横向代码重用的概念,并且不希望必然地继承抽象类。我不明白的是:使用特征和使用界面之间的关键区别是什么?
我曾试着搜索过一篇像样的博客文章或文章,解释什么时候使用其中一种或另一种,但到目前为止,我找到的例子似乎非常相似,甚至完全相同。
当前回答
我认为trait对于创建包含方法的类是有用的,这些方法可以被用作几个不同类的方法。
例如:
trait ToolKit
{
public $errors = array();
public function error($msg)
{
$this->errors[] = $msg;
return false;
}
}
您可以在任何使用此特性的类中拥有并使用此“error”方法。
class Something
{
use Toolkit;
public function do_something($zipcode)
{
if (preg_match('/^[0-9]{5}$/', $zipcode) !== 1)
return $this->error('Invalid zipcode.');
// do something here
}
}
而对于接口,你只能声明方法签名,而不能声明它的函数代码。此外,要使用接口,您需要遵循层次结构,使用实现。但特质却不是这样。
这是完全不同的!
其他回答
该特征与我们可以用于多重继承目的和代码可重用性的类相同。
我们可以在类中使用trait,也可以在同一个类中使用'use keyword'来使用多个trait。
接口用于代码可重用性与特性相同
接口是扩展多个接口,所以我们可以解决多个继承问题,但当我们实现接口时,我们应该在类中创建所有的方法。 欲了解更多信息,请点击以下链接:
http://php.net/manual/en/language.oop5.traits.php http://php.net/manual/en/language.oop5.interfaces.php
描述trait的一个常用比喻是:trait是带有实现的接口。
在大多数情况下,这是一种很好的思考方式,但两者之间存在一些微妙的差异。
首先,instanceof操作符不能与trait一起工作(即,trait不是一个真正的对象),因此不能使用它来查看一个类是否具有某个trait(或者查看两个本来不相关的类是否共享一个trait)。这就是他们所说的横向代码重用的结构。
现在PHP中有一些函数可以让您获得类使用的所有特征的列表,但是特征继承意味着您需要进行递归检查,以可靠地检查类在某个时刻是否具有特定的特征(PHP doco页面上有示例代码)。不过,它当然不像instanceof那样简单干净,以我之见,这个特性会让PHP变得更好。
而且,抽象类仍然是类,因此它们不能解决与多重继承相关的代码重用问题。请记住,您只能扩展一个类(真实的或抽象的),但可以实现多个接口。
我发现trait和接口非常适合一起使用来创建伪多重继承。例如:
class SlidingDoor extends Door implements IKeyed
{
use KeyedTrait;
[...] // Generally not a lot else goes here since it's all in the trait
}
这样做意味着您可以使用instanceof来确定特定的Door对象是否是Keyed的,您知道您将得到一组一致的方法,等等,并且所有的代码在所有使用KeyedTrait的类中都位于一个位置。
接口是一种契约,它表明“这个对象能够做这件事”,而trait则赋予对象做这件事的能力。
trait本质上是在类之间“复制和粘贴”代码的一种方式。
试着阅读这篇文章,PHP的特点是什么?
对于初学者来说,上面的答案可能很难,下面是最简单的理解方法:
特征
trait SayWorld {
public function sayHello() {
echo 'World!';
}
}
所以如果你想在其他类中使用sayHello函数,而不需要重新创建整个函数,你可以使用trait,
class MyClass{
use SayWorld;
}
$o = new MyClass();
$o->sayHello();
酷吧!
不只是函数,你可以使用trait中的任何东西(function, variables, const…)你也可以使用多个trait: SayWorld, AnotherTraits;
接口
interface SayWorld {
public function sayHello();
}
class MyClass implements SayWorld {
public function sayHello() {
echo 'World!';
}
}
因此,这就是接口与特征的不同之处:您必须在实现的类中重新创建接口中的所有内容。接口没有实现,接口只能有函数和常量,不能有变量。
我希望这能有所帮助!
基本上,您可以将trait视为代码的自动“复制-粘贴”。
使用trait是很危险的,因为在执行前我们无法知道它的作用。
然而,由于缺乏遗传等限制,性状更加灵活。
trait在注入方法时很有用,它可以检查类中是否存在另一个方法或属性。这是一篇不错的文章(但是是法语,抱歉)。
对于能读法语的人来说,GNU/Linux杂志HS 54有一篇关于这个主题的文章。