从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时同名方法冲突的处理方法
# 多个
# 多态
# 复用
# 如图
# 面向对象
# 类中
# 我们可以
# 这两个
# 可以看到
# 实现了
# 是在
# 子类
# 是一样的
# 要有
# 也能
# 两种
# 说到
# 要在
# 比较好
# 这个时候
相关文章:
宝塔面板创建网站无法访问?如何快速排查修复?
如何通过智能用户系统一键生成高效建站方案?
网站制作的步骤包括,正确网址格式怎么写?
制作证书网站有哪些,全国城建培训中心证书查询官网?
网站制作软件免费下载安装,有哪些免费下载的软件网站?
如何快速搭建高效服务器建站系统?
怎么用手机制作网站链接,dw怎么把手机适应页面变成网页?
如何高效完成独享虚拟主机建站?
建站之星后台密码遗忘如何找回?
设计网站制作公司有哪些,制作网页教程?
上海网站制作开发公司,上海买房比较好的网站有哪些?
建站10G流量真的够用吗?如何应对访问高峰?
网站制作报价单模板图片,小松挖机官方网站报价?
宝塔建站助手安装配置与建站模板使用全流程解析
网站制作公司广州有几家,广州尚艺美发学校网站是多少?
微信推文制作网站有哪些,怎么做微信推文,急?
小型网站建站如何选择虚拟主机?
学校为何禁止电信移动建设网站?
专业网站制作企业网站,如何制作一个企业网站,建设网站的基本步骤有哪些?
建站之星展会模版如何一键下载生成?
建站之星代理费用多少?最新价格详情介绍
建站主机选择指南:服务器配置与SEO优化实战技巧
制作网站的过程怎么写,用凡科建站如何制作自己的网站?
如何撰写建站申请书?关键要点有哪些?
婚礼视频制作网站,学习*后期制作的网站有哪些?
潮流网站制作头像软件下载,适合母子的网名有哪些?
企业在线网站设计制作流程,想建设一个属于自己的企业网站,该如何去做?
如何自定义建站之星模板颜色并下载新样式?
在线制作视频网站免费,都有哪些好的动漫网站?
C++用Dijkstra(迪杰斯特拉)算法求最短路径
视频网站app制作软件,有什么好的视频聊天网站或者软件?
网站按钮制作软件,如何实现网页中按钮的自动点击?
安云自助建站系统如何快速提升SEO排名?
高防服务器租用首荐平台,企业级优惠套餐快速部署
javascript基本数据类型及类型检测常用方法小结
建站之星ASP如何实现CMS高效搭建与安全管理?
如何在宝塔面板创建新站点?
建站之星备案是否影响网站上线时间?
太平洋网站制作公司,网络用语太平洋是什么意思?
七夕网站制作视频,七夕大促活动怎么报名?
Swift开发中switch语句值绑定模式
高性能网站服务器部署指南:稳定运行与安全配置优化方案
建站之星如何防范黑客攻击与数据泄露?
建站之星3.0如何解决常见操作问题?
如何在Ubuntu系统下快速搭建WordPress个人网站?
建站之星安装后如何自定义网站颜色与字体?
如何通过二级域名建站提升品牌影响力?
沈阳个人网站制作公司,哪个网站能考到沈阳事业编招聘的信息?
建站之星微信建站一键生成小程序+多端营销系统
如何通过.red域名打造高辨识度品牌网站?
*请认真填写需求信息,我们会在24小时内与您取得联系。