从PHP的5.4.0版本开始,PHP提供了一种全新的代码复用的概念,那就是Trait。Trait其字面意思是”特性”、”特点”,我们可以理解为,使用Trait关键字,可以为PHP中的类添加新的特性。

熟悉面向对象的都知道,软件开发中常用的代码复用有继承和多态两种方式。在PHP中,只能实现单继承。而Trait则避免了这点。下面通过简单的额例子来进行对比说明。
1. 继承 VS 多态 VS Trait
现在有Publish.php和Answer.php这两个类。要在其中添加LOG功能,记录类内部的动作。有以下几种方案:
继承
多态
Trait
1.1. 继承
如图:
代码结构如下:
// Log.php
<?php
Class Log
{
public function startLog()
{
// echo ...
}
public function endLog()
{
// echo ...
}
}
// Publish.php
<?php
Class Publish extends Log
{
} // Answer.php
<?php
Class Answer extends Log
{
}
可以看到继承的确满足了要求。但这却违背了面向对象的原则。而发布(Publish)和回答(Answer)这样的操作和日志(Log)之间的关系并不是子类与父类的关系。所以不推荐这样使用。
1.2. 多态
如图:
实现代码:
// Log.php
<?php
Interface Log
{
public function startLog();
public function endLog();
}
// Publish.php
<?php
Class Publish implements Log
{
public function startLog()
{
// TODO: Implement startLog() method.
}
public function endLog()
{
// TODO: Implement endLog() method.
}
}
// Answer.php
<?php
Class Answer implements Log
{
public function startLog()
{
// TODO: Implement startLog() method.
}
public function endLog()
{
// TODO: Implement endLog() method.
}
}
记录日志的操作应该都是一样的,因此,发布(Publish)和回答(Answer)动作中的日志记录实现也是一样的。很明显,这违背了DRY(Don't Repeat Yourself)原则。所以是不推荐这样实现的。
1.3. Trait
如图:
实现代码如下:
// Log.php
<?php
trait Log{
public function startLog() {
// echo ..
}
public function endLog() {
// echo ..
}
}
// Publish.php
<?php
class Publish {
use Log;
}
$publish = new Publish();
$publish->startLog();
$publish->endLog();
// Answer.php
<?php
class Answer {
use Log;
}
$answer = new Answer();
$answer->startLog();
$answer->endLog();
可以看到,我们在没有增加代码复杂的情况下,实现了代码的复用。
1.4. 结论
继承的方式虽然也能解决问题,但其思路违背了面向对象的原则,显得很粗暴;多态方式也可行,但不符合软件开发中的DRY原则,增加了维护成本。而Trait方式则避免了上述的不足之处,相对优雅的实现了代码的复用。
2. Trait的作用域
了解了Trait的好处,我们还需要了解其实现中的规则,先来说一下作用域。这个比较好证明,实现代码如下:
<?php
class Publish {
use Log;
public function doPublish() {
$this->publicF();
$this->protectF();
$this->privateF();
}
}
$publish = new Publish();
$publish->doPublish();
执行上述代码输出结果如下:
public function
protected function
private function
可以发现,Trait的作用域在引用该Trait类的内部是都可见的。可以理解为use关键字将Trait的实现代码Copy了一份到引用该Trait的类中。
3. Trait中属性的优先级
说到优先级,就必须要有一个对比的参照物,这里的参照对象时引用Trait的类及其父类。
通过以下的代码来证明Trait应用中的属性的优先级:
<?php
trait Log
{
public function publicF()
{
echo __METHOD__ . ' public function' . PHP_EOL;
}
protected function protectF()
{
echo __METHOD__ . ' protected function' . PHP_EOL;
}
}
class Question
{
public function publicF()
{
echo __METHOD__ . ' public function' . PHP_EOL;
}
protected function protectF()
{
echo __METHOD__ . ' protected function' . PHP_EOL;
}
}
class Publish extends Question
{
use Log;
public function publicF()
{
echo __METHOD__ . ' public function' . PHP_EOL;
}
public function doPublish()
{
$this->publicF();
$this->protectF();
}
}
$publish = new Publish();
$publish->doPublish();
上述代码的输出结果如下:
Publish::publicF public function
Log::protectF protected function
通过上面的例子,可以总结出Trait应用中的优先级如下:
1.来自当前类的成员覆盖了 trait 的方法
2.trait 覆盖了被继承的方法
类成员优先级为:当前类>Trait>父类
4. Insteadof和As关键字
在一个类中,可以引用多个Trait,如下:
<?php
trait Log
{
public function startLog()
{
echo __METHOD__ . ' public function' . PHP_EOL;
}
protected function endLog()
{
echo __METHOD__ . ' protected function' . PHP_EOL;
}
}
trait Check
{
public function parameterCheck($parameters) {
// do sth
}
}
class Publish extends Question
{
use Log,Check;
public function doPublish($para) {
$this->startLog();
$this->parameterCheck($para);
$this->endLog();
}
}
通过上面的方式,我们可以在一个类中引用多个Trait。引用多个Trait的时候,就容易出问题了,最常见的问题就是两个Trait中如果出现了同名的属性或者方法该怎么办呢?这个时候就需要用到Insteadof 和 as 这两个关键字了.请看如下实现代码:
<?php
trait Log
{
public function parameterCheck($parameters)
{
echo __METHOD__ . ' parameter check' . $parameters . PHP_EOL;
}
public function startLog()
{
echo __METHOD__ . ' public function' . PHP_EOL;
}
}
trait Check
{
public function parameterCheck($parameters)
{
echo __METHOD__ . ' parameter check' . $parameters . PHP_EOL;
}
public function startLog()
{
echo __METHOD__ . ' public function' . PHP_EOL;
}
}
class Publish
{
use Check, Log {
Check::parameterCheck insteadof Log;
Log::startLog insteadof Check;
Check::startLog as csl;
}
public function doPublish()
{
$this->startLog();
$this->parameterCheck('params');
$this->csl();
}
}
$publish = new Publish();
$publish->doPublish();
执行上述代码,输出结果如下:
Log::startLog public function
Check::parameterCheck parameter checkparams
Check::startLog public function
就如字面意思一般,insteadof关键字用前者取代了后者,as 关键字给被取代的方法起了一个别名。
在引用Trait时,使用了use关键字,use关键字也用来引用命名空间。两者的区别在于,引用Trait时是在class内部使用的。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# PHP
# Trait
# 浅谈PHP中的Trait使用方法
# PHP中trait使用方法详细介绍
# PHP中的Trait 特性及作用
# PHP中的traits简单使用实例
# PHP中trait的使用和同时引入多个trait时同名方法冲突的处理方法
# 多个
# 多态
# 复用
# 如图
# 面向对象
# 类中
# 我们可以
# 这两个
# 可以看到
# 实现了
# 是在
# 子类
# 是一样的
# 要有
# 也能
# 两种
# 说到
# 要在
# 比较好
# 这个时候
相关文章:
整蛊网站制作软件,手机不停的收到各种网站的验证码短信,是手机病毒还是人为恶搞?有这种手机病毒吗?
如何用虚拟主机快速搭建网站?详细步骤解析
广州商城建站系统开发成本与周期如何控制?
如何通过服务器快速搭建网站?完整步骤解析
成都网站制作报价公司,成都工业用气开户费用?
c# Task.Yield 的作用是什么 它和Task.Delay(1)有区别吗
如何选择高性价比服务器搭建个人网站?
如何在Golang中实现微服务服务拆分_Golang微服务拆分与接口管理方法
义乌企业网站制作公司,请问义乌比较好的批发小商品的网站是什么?
常州企业建站如何选择最佳模板?
如何通过智能用户系统一键生成高效建站方案?
高防网站服务器:DDoS防御与BGP线路的AI智能防护方案
广州美橙建站如何快速搭建多端合一网站?
linux top下的 minerd 木马清除方法
如何选择美橙互联多站合一建站方案?
如何批量查询域名的建站时间记录?
如何确保西部建站助手FTP传输的安全性?
如何获取免费开源的自助建站系统源码?
如何零基础开发自助建站系统?完整教程解析
广平建站公司哪家专业可靠?如何选择?
车管所网站制作流程,交警当场开简易程序处罚决定书,在交警网站查询不到怎么办?
建站主机SSH密钥生成步骤及常见问题解答?
外汇网站制作流程,如何在工商银行网站上做外汇买卖?
建站ABC备案流程中有哪些关键注意事项?
建站之星代理如何获取技术支持?
如何生成腾讯云建站专用兑换码?
制作网页的网站有哪些,电脑上怎么做网页?
网页设计与网站制作内容,怎样注册网站?
高防服务器租用首荐平台,企业级优惠套餐快速部署
如何在VPS电脑上快速搭建网站?
电脑免费海报制作网站推荐,招聘海报哪个网站多?
制作网站的过程怎么写,用凡科建站如何制作自己的网站?
盐城做公司网站,江苏电子版退休证办理流程?
公司网站建设制作费用,想建设一个属于自己的企业网站,该如何去做?
高防服务器租用指南:配置选择与快速部署攻略
建站之星安装需要哪些步骤及注意事项?
如何在香港免费服务器上快速搭建网站?
广州建站公司哪家好?十大优质服务商推荐
建站IDE高效指南:快速搭建+SEO优化+自适应模板全解析
如何通过FTP服务器快速搭建网站?
公众号网站制作网页,微信公众号怎么制作?
西安制作网站公司有哪些,西安货运司机用的最多的app或者网站是什么?
如何通过商城自助建站源码实现零基础高效建站?
天津个人网站制作公司,天津网约车驾驶员从业资格证官网?
成都品牌网站制作公司,成都营业执照年报网上怎么办理?
如何通过山东自助建站平台快速注册域名?
建设网站制作价格,怎样建立自己的公司网站?
如何在IIS中新建站点并配置端口与IP地址?
南平网站制作公司,2025年南平市事业单位报名时间?
建站之星后台管理如何实现高效配置?
*请认真填写需求信息,我们会在24小时内与您取得联系。