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

熟悉面向对象的都知道,软件开发中常用的代码复用有继承和多态两种方式。在PHP中,只能实现单继承。而Trait则避免了这点。下面通过简单的额例子来进行对比说明。
1. 继承 VS 多态 VS Trait
现在有Publish.php和Answer.php这两个类。要在其中添加LOG功能,记录类内部的动作。有以下几种方案:
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应用中的优先级如下:
来自当前类的成员覆盖了 trait 的方法
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
# 5.4
# 代码复用
# Trait
# Trait新特性
# PHP 实现代码复用的一个方法 traits新特性
# PHP中的Trait 特性及作用
# PHP中trait使用方法详细介绍
# 浅谈PHP中的Trait使用方法
# PHP中Trait及其应用详解
# 简单谈谈PHP中的trait
# PHP中的traits实现代码复用使用实例
# PHP Trait代码复用类与多继承实现方法详解
# 详解PHP神奇又有用的Trait
# PHP的Trait机制原理与用法分析
# PHP中用Trait封装单例模式的实现
# PHP Trait功能与用法实例分析
# 多个
# 多态
# 复用
# 如图
# 面向对象
# 类中
# 我们可以
# 这两个
# 可以看到
# 实现了
# 是在
# 子类
# 是一样的
# 要有
# 也能
# 两种
# 说到
# 要在
# 比较好
# 这个时候
相关文章:
如何快速查询网址的建站时间与历史轨迹?
如何在云服务器上快速搭建个人网站?
建站VPS配置与SEO优化指南:关键词排名提升策略
香港服务器建站指南:免备案优势与SEO优化技巧全解析
如何在企业微信快速生成手机电脑官网?
个人摄影网站制作流程,摄影爱好者都去什么网站?
网站设计制作公司地址,网站建设比较好的公司都有哪些?
公众号网站制作网页,微信公众号怎么制作?
建站之星备案是否影响网站上线时间?
简单实现Android文件上传
如何在云主机快速搭建网站站点?
魔毅自助建站系统:模板定制与SEO优化一键生成指南
香港服务器网站推广:SEO优化与外贸独立站搭建策略
如何通过可视化优化提升建站效果?
济南网站制作的价格,历城一职专官方网站?
建站主机服务器选型指南与性能优化方案解析
c++怎么编写动态链接库dll_c++ __declspec(dllexport)导出与调用【方法】
如何在云虚拟主机上快速搭建个人网站?
表情包在线制作网站免费,表情包怎么弄?
简历在线制作网站免费版,如何创建个人简历?
建站之星代理费用多少?最新价格详情介绍
建站之星安装步骤有哪些常见问题?
建站主机功能解析:服务器选择与快速搭建指南
模具网站制作流程,如何找模具客户?
如何在阿里云域名上完成建站全流程?
Python多线程使用规范_线程安全解析【教程】
大同网页,大同瑞慈医院官网?
如何在IIS管理器中快速创建并配置网站?
黑客如何通过漏洞一步步攻陷网站服务器?
湖北网站制作公司有哪些,湖北清能集团官网?
网站制作说明怎么写,简述网页设计的流程并说明原因?
如何在Mac上搭建Golang开发环境_使用Homebrew安装和管理Go版本
,购物网站怎么盈利呢?
如何在Windows环境下新建FTP站点并设置权限?
视频网站app制作软件,有什么好的视频聊天网站或者软件?
相亲简历制作网站推荐大全,新相亲大会主持人小萍萍资料?
如何用花生壳三步快速搭建专属网站?
如何通过虚拟主机空间快速建站?
学校免费自助建站系统:智能生成+拖拽设计+多端适配
微网站制作教程,不会写代码,不会编程,怎么样建自己的网站?
非常酷的网站设计制作软件,酷培ai教育官方网站?
如何用AWS免费套餐快速搭建高效网站?
赚钱网站制作软件,建一个网站怎样才能赚钱?是如何盈利的?
如何配置WinSCP新建站点的密钥验证步骤?
如何选择美橙互联多站合一建站方案?
美食网站链接制作教程视频,哪个教做美食的网站比较专业点?
高防服务器租用指南:配置选择与快速部署攻略
陕西网站制作公司有哪些,陕西凌云电器有限公司官网?
如何用wdcp快速搭建高效网站?
制作网站公司那家好,网络公司是做什么的?
*请认真填写需求信息,我们会在24小时内与您取得联系。