zoukankan      html  css  js  c++  java
  • PHP之Trait详解

    php从以前到现在一直都是单继承的语言,无法同时从两个基类中继承属性和方法,为了解决这个问题,php出了Trait这个特性

    用法:通过在类中使用use 关键字,声明要组合的Trait名称,具体的Trait的声明使用Trait关键词,Trait不能实例化

    与普通类的异同:

    相同:

    trait 能够像普通的类一样定义属性,方法(包含抽象的、静态的、抽象的);
    trait 引入到基类里面,其子类里面也能访问trait里面的属性和方法。
    不同:

    trait不用实例化就能访问定义的普通方法以及属性。
    trait里面不能定义构造函数。

    如下代码实例:

    <?php
    trait Dog{
        public $name="dog";
        public function bark(){
            echo "This is dog";
        }
    }
    class Animal{
        public function eat(){
            echo "This is animal eat";
        }
    }
    class Cat extends Animal{
        use Dog;
        public function drive(){
            echo "This is cat drive";
        }
    }
    $cat = new Cat();
    $cat->drive();
    echo "<br/>";
    $cat->eat();
    echo "<br/>";
    $cat->bark();
    ?>

    将会如下输出

    优先级

    从基类继承的成员会被 trait 插入的成员所覆盖。优先顺序是来自当前类的成员覆盖了 trait 的方法,而 trait 则覆盖了被继承的方法。

    从基类继承的成员被插入的 SayWorld Trait 中的 MyHelloWorld 方法所覆盖。其行为 MyHelloWorld 类中定义的方法一致。优先顺序是当前类中的方法会覆盖 trait 方法,而 trait 方法又覆盖了基类中的方法。

    <?php
    class Base {
        public function sayHello() {
            echo 'Hello ';
        }
    }
    
    trait SayWorld {
        public function sayHello() {
            parent::sayHello();
            echo 'World!';
        }
    }
    
    class MyHelloWorld extends Base {
        use SayWorld;
    }
    
    $o = new MyHelloWorld();
    $o->sayHello();
    ?>

    以上例程会输出:

    Hello World!

    再测试Trait、基类和本类对同名属性或方法的处理,如下代码

    <?php
    trait Dog{
        public $name="dog";
        public function drive(){
            echo "This is dog drive";
        }
        public function eat(){
            echo "This is dog eat";
        }
    }
    
    class Animal{
        public function drive(){
            echo "This is animal drive";
        }
        public function eat(){
            echo "This is animal eat";
        }
    }
    
    class Cat extends Animal{
    use Dog;
    public function drive(){ echo "This is cat drive"; } } $cat = new Cat(); $cat->drive(); echo "<br/>"; $cat->eat(); ?>

    如下显示

    所以:Trait中的方法会覆盖 基类中的同名方法,而本类会覆盖Trait中同名方法
    注意点:当trait定义了属性后,类就不能定义同样名称的属性,否则会产生 fatal error,除非是设置成相同可见度、相同默认值。不过在php7之前,即使这样设置,还是会产生E_STRICT 的提醒 ,有种情况例外:属性是兼容的(同样的访问可见度、初始默认值)。 在 PHP 7.0 之前,属性是兼容的,则会有 E_STRICT 的提醒。。

     

    一个类可以组合多个Trait,通过逗号相隔,如下

    use trait1,trait2

    当不同的trait中,却有着同名的方法或属性,会产生冲突,可以使用insteadof或 as进行解决,insteadof 是进行替代,而as是给它取别名
    如下实例:

    <?php
    trait trait1{
        public function eat(){
            echo "This is trait1 eat";
        }
        public function drive(){
            echo "This is trait1 drive";
        }
    }
    trait trait2{
        public function eat(){
            echo "This is trait2 eat";
        }
        public function drive(){
            echo "This is trait2 drive";
        }
    }
    class cat{
        use trait1,trait2{
            trait1::eat insteadof trait2;
            trait1::drive insteadof trait2;
        }
    }
    class dog{
        use trait1,trait2{
            trait1::eat insteadof trait2;
            trait1::drive insteadof trait2;
            trait2::eat as eaten;
            trait2::drive as driven;
        }
    }
    $cat = new cat();
    $cat->eat();
    echo "<br/>";
    $cat->drive();
    echo "<br/>";
    echo "<br/>";
    echo "<br/>";
    $dog = new dog();
    $dog->eat();
    echo "<br/>";
    $dog->drive();
    echo "<br/>";
    $dog->eaten();
    echo "<br/>";
    $dog->driven();
    ?>

    输出如下

    as 还可以修改方法的访问控制

    <?php
    trait Animal{
        public function eat(){
            echo "This is Animal eat";
        }
    }
    
    class Dog{
        use Animal{
            eat as protected;
        }
    }
    class Cat{
        use Animal{
            Animal::eat as private eaten;
        }
    }
    $dog = new Dog();
    $dog->eat();//报错,因为已经把eat改成了保护
    
    $cat = new Cat();
    $cat->eat();//正常运行,不会修改原先的访问控制
    $cat->eaten();//报错,已经改成了私有的访问控制
    ?>

    Trait也可以互相组合,还可以使用抽象方法,静态属性,静态方法等,实例如下

    <?php
    trait Cat{
        public function eat(){
            echo "This is Cat eat";
        }
    }
    
    trait Dog{
        use Cat;
        public function drive(){
            echo "This is Dog drive";
        }
        abstract public function getName();
        
        public function test(){
            static $num=0;
            $num++;
            echo $num;
        }
        
        public static function say(){
            echo "This is Dog say";
        }
    }
    class animal{
        use Dog;
        public function getName(){
            echo "This is animal name";
        }
    }
    
    $animal = new animal();
    $animal->getName();
    echo "<br/>";
    $animal->eat();
    echo "<br/>";
    $animal->drive();
    echo "<br/>";
    $animal::say();
    echo "<br/>";
    $animal->test();
    echo "<br/>";
    $animal->test();
    ?>

    输出如下

  • 相关阅读:
    跨域资源共享 CORS 详解
    C# 每月第一天和最后一天
    jexus http to https
    ASP.NET MVC 路由学习
    jexus
    centos7 添加开机启动项
    jexus docker
    HTTP2.0新特性
    jexus配置支持Owin
    Autofac Named命名和Key Service服务
  • 原文地址:https://www.cnblogs.com/xiangshihua/p/13282170.html
Copyright © 2011-2022 走看看