• PHP 的 self 关键字用法



    之前有人询问 self 关键字的用法,答案是比较明显的:静态成员函数内不能用 this 调用非成员函数,但可以用 self 调用静态成员函数/变量/常量;其他成员函数可以用 self 调用静态成员函数以及非静态成员函数。随着讨论的深入,发现 self 并没有那么简单。鉴于此,本文先对几个关键字做对比和区分,再总结 self 的用法。

    parentstatic 以及 this 的区别

    要想将彻底搞懂 self ,要与 parentstatic 以及 this 区分开。以下分别做对比。

    parent

    selfparent 的区分比较容易: parent 引用父类/基类被隐盖的方法(或变量), self则引用自身方法(或变量)。例如构造函数中调用父类构造函数:

    class Base {
        public function __construct() {
            echo "Base contructor!", PHP_EOL;
        }
    }
    
    class Child {
        public function __construct() {
            parent::__construct();
            echo "Child contructor!", PHP_EOL;
        }
    }
    
    new Child;
    // 输出:
    // Base contructor!
    // Child contructor!

    static

    static 常规用途是修饰函数或变量使其成为类函数和类变量,也可以修饰函数内变量延长其生命周期至整个应用程序的生命周期。但是其与 self 关联上是PHP 5.3以来引入的新用途:静态延迟绑定。

    有了 static 的静态延迟绑定功能,可以在运行时动态确定归属的类。例如:

    class Base {
        public function __construct() {
            echo "Base constructor!", PHP_EOL;
        }
    
        public static function getSelf() {
            return new self();
        }
    
        public static function getInstance() {
            return new static();
        }
    
        public function selfFoo() {
            return self::foo();
        }
    
        public function staticFoo() {
            return static::foo();
        }
    
        public function thisFoo() {
            return $this->foo();
        }
    
        public function foo() {
            echo  "Base Foo!", PHP_EOL;
        }
    }
    
    class Child extends Base {
        public function __construct() {
            echo "Child constructor!", PHP_EOL;
        }
    
        public function foo() {
            echo "Child Foo!", PHP_EOL;
        }
    }
    
    $base = Child::getSelf();
    $child = Child::getInstance();
    
    $child->selfFoo();
    $child->staticFoo();
    $child->thisFoo();

    程序输出结果如下:

    Base constructor!
    Child constructor!
    Base Foo!
    Child Foo!
    Child Foo!

    在函数引用上, selfstatic 的区别是:对于静态成员函数, self 指向代码当前类, static 指向调用类;对于非静态成员函数, self 抑制多态,指向当前类的成员函数, static 等同于 this ,动态指向调用类的函数。

    parentselfstatic 三个关键字联合在一起看挺有意思,分别指向父类、当前类、子类,有点“过去、现在、未来”的味道。

    this

    selfthis 是被讨论最多,也是最容易引起误用的组合。两者的主要区别如下:

    1. this 不能用在静态成员函数中, self 可以;
    2. 对静态成员函数/变量的访问, 建议 用 self ,不要用 $this::$this-> 的形式;
    3. 对非静态成员变量的访问,不能用 self ,只能用 this ;
    4. this 要在对象已经实例化的情况下使用, self 没有此限制;
    5. 在非静态成员函数内使用, self 抑制多态行为,引用当前类的函数;而 this 引用调用类的重写(override)函数(如果有的话)。

    self 的用途

    看完与上述三个关键字的区别, self 的用途是不是呼之即出?一句话总结,那就是: self总是指向“当前类(及类实例)”。详细说则是:

    1. 替代类名,引用当前类的静态成员变量和静态函数;
    2. 抑制多态行为,引用当前类的函数而非子类中覆盖的实现;

    槽点

    1. 这几个关键字中,只有 this 要加 $ 符号且必须加,强迫症表示很难受;
    2. 静态成员函数中不能通过 $this-> 调用非静态成员函数,但是可以通过 self:: 调用,且在调用函数中未使用 $this-> 的情况下还能顺畅运行。此行为貌似在不同PHP版本中表现不同,在当前的7.3中ok;
    3. 在静态函数和非静态函数中输出 self ,猜猜结果是什么?都是 string(4) "self" ,迷之输出;
    4. return $this instanceof static::class; 会有语法错误,但是以下两种写法就正常:
      $class = static::class;
      return $this instanceof $class;
      // 或者这样:
      return $this instanceof static;

     很多PHPer在进阶的时候总会遇到一些问题和瓶颈,业务代码写多了没有方向感,不知道该从那里入手去提升,对此我整理了一些资料,包括但不限于:分布式架构、高可扩展、高性能、高并发、服务器性能调优、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql优化、shell脚本、Docker、微服务、Nginx等多个知识点高级进阶干货需要的可以免费分享给大家,需要的加群(点击→)677079770

  • 相关阅读:
    非阻塞式线程安全列表-ConcurrentLinkedDeque
    计数器
    Linux 查看服务器内存使用情况
    Oracle to_date, to_timestamp
    Hibernate session.flush() 使用
    阿里规约认证(题库附答案)
    多态性
    Java数据类型
    String类特点分析
    数组的定义
  • 原文地址:https://www.cnblogs.com/a609251438/p/11938922.html
Copyright © 2020-2023  润新知