PHP中Trait的使用

  • A+
所属分类:PHP

面向对象编程的三大特性是:抽象,继承,多态。我们都知道,PHP是一种单继承的语言,那么如何在PHP中如何实现多继承呢?

我们先来考虑这样一个需求: 我们的项目是多模块,多平台开发。我们会使用MVC分层,将操作数据库的方法放在M层,接收参数验证参数放在C层,V层用来渲染页面。那么有这样这样一组方法,需要在M层和C层都需要调用,怎么写这个方法呢?

方法①: 我们将方法写在 M层和C层的基类中, 分别继承。 虽然实现了需求,但是M层和C层的基类中还是会存在重复代码

方法②:    我们将方法写在公共的一个类,使用静态方法或者单例模式调用。 这样也可以实现需求,并且看起来没有什么问题。

那么我们再来思考一个问题:我们的项目是多模块,多后台登录,对于登录来讲, 两个模块中的控制器内容应该是相似的,因为他们的工作是相同的,接收登录的用户名密码并验证参数,交给模型处理并返回登录结果。那如何封装代码呢?

有的同学会说: 我们可以建一个登录的基类,两个模块分别继承就可以拥有登录的功能,并且维护起来也十分方便!

这样也实现了需求,但是缺点在于增加继承的深度,而且登录的父类还要去继承控制器的基类,继承的层度只会越来越深。

我们在来思考:在项目中我们的数据库中使用软删除(即删除是在数据库表中加入字段标记,而不是物理删除),我们的模型去继承软删除的基类就可以实现此方法呢? 假如现在我们想要在一些重要的模型中记录增删改查的记录,通过在模型的钩子(添加后,删除后,修改后)内记录日志,难道我们的模型还要去记录日志模型吗? 假如已经继承了软删除,如何再继承日志模型呢?

在php5.4 以后,php内加入了 trait 来解决多继承的问题。

trait 的目的在于减少代码的重复,增加复用性。

用法:使用trait 声明, 在要使用的类中使用关键字use ,不能被初始化。

trait LoginTrait
{
    public function loginAction()
    {
        //接收参数 验证参数
        //调用business
        echo 'success';
        echo '<br/>';
        //返回结果
    }
}

class AdminLogin
{
    use LoginTrait;
}

class ShopLogin
{
    use LoginTrait;
}

我们需要注意的是 trait 的优先级: 自身方法>trait的方法>继承的方法

class Login
{
    public function LoginAction()
    {
        echo 'success parent';
    }
}

trait LoginTrait
{
    public function loginAction()
    {
        //接收参数 验证参数
        //调用business
        echo 'success';
        echo '<br/>';
        //返回结果
    }
}

class AdminLogin extends Login
{
    use LoginTrait;

    public function loginAction()
    {
        echo 'success self';
    }
}

(new AdminLogin())->loginAction();

输出结果:success self

那么一个类是否可以use 多个 trait 呢? 答案是 可以的, 但是要注意的是,当use 的 多个 trait 中 有相同的方法时,会引起冲突,需要在子类中使用 insteadof 或者 as 解决。

trait LoginTrait2
{
    public function loginAction()
    {
        //接收参数 验证参数
        //调用business
        echo 'success2';
        echo '<br/>';
        //返回结果
    }
}

class AdminLogin extends Login
{
    use LoginTrait, LoginTrait2{
        LoginTrait2::loginAction insteadof LoginTrait;
        LoginTrait::loginAction as loginAction2;
    }

}

(new AdminLogin())->loginAction();
(new AdminLogin())->loginAction2();

输入结果: success2;  success

Trait的应用:

如我们前面举到的例子: 软删除,日志模型,另外还有如:缓存模型、复用单例类的方法、复用多平台的控制器内容等等。

avatar

发表评论

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen: