当前位置:首页 > PHP教程 > php高级应用 > 列表

分分钟搞定PHP的self关键字

发布:smiling 来源: PHP粉丝网  添加日期:2022-05-26 08:08:36 浏览: 评论:0 

静态成员函数内不能用this调用非成员函数,但可以用self调用静态成员函数/变量/常量;

其他成员函数可以用self调用静态成员函数以及非静态成员函数。

随着讨论的深入,发现self并没有那么简单。鉴于此,本文先对几个关键字做对比和区分,再总结self的用法。

与parent、static以及this的区别

要想将彻底搞懂self,要与parent、static、this区分开。

静态成员函数内不能用this调用非成员函数,但可以用self调用静态成员函数/变量/常量;

其他成员函数可以用self调用静态成员函数以及非静态成员函数。

随着讨论的深入,发现self并没有那么简单。鉴于此,本文先对几个关键字做对比和区分,再总结self的用法。

与parent、static以及this的区别

要想将彻底搞懂self,要与parent、static、this区分开。

  1. class Base { 
  2.  
  3.     public function __construct() { 
  4.  
  5.         echo "Base contructor!", PHP_EOL; 
  6.  
  7.     } 
  8.  
  9.  
  10.     
  11.  
  12. class Child { 
  13.  
  14.     public function __construct() { 
  15.  
  16.         parent::__construct(); 
  17.  
  18.         echo "Child contructor!", PHP_EOL; 
  19.  
  20.     } 
  21.  
  22.  
  23.     
  24.  
  25. new Child; 
  26.  
  27. // 输出: 
  28.  
  29. // Base contructor! 
  30.  
  31. // Child contructor! 

static

static常规用途是修饰函数或变量使其成为类函数和类变量,也可以修饰函数内变量延长其生命周期至整个应用程序的生命周期。

但是其与self关联上是PHP 5.3以来引入的新用途:静态延迟绑定。

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

例如:

  1. class Base { 
  2.  
  3.     public function __construct() { 
  4.  
  5.         echo "Base constructor!", PHP_EOL; 
  6.  
  7.     } 
  8.  
  9.     
  10.  
  11.     public static function getSelf() { 
  12.  
  13.         return new self(); 
  14.  
  15.     } 
  16.  
  17.     
  18.  
  19.     public static function getInstance() { 
  20.  
  21.         return new static(); 
  22.  
  23.     } 
  24.  
  25.     
  26.  
  27.     public function selfFoo() { 
  28.  
  29.         return self::foo(); 
  30.  
  31.     } 
  32.  
  33.     
  34.  
  35.     public function staticFoo() { 
  36.  
  37.         return static::foo(); 
  38.  
  39.     } 
  40.  
  41.     
  42.  
  43.     public function thisFoo() { 
  44.  
  45.         return $this->foo(); 
  46.  
  47.     } 
  48.  
  49.     
  50.  
  51.     public function foo() { 
  52.  
  53.         echo  "Base Foo!", PHP_EOL; 
  54.  
  55.     } 
  56.  
  57.  
  58.     
  59.  
  60. class Child extends Base { 
  61.  
  62.     public function __construct() { 
  63.  
  64.         echo "Child constructor!", PHP_EOL; 
  65.  
  66.     } 
  67.  
  68.     
  69.  
  70.     public function foo() { 
  71.  
  72.         echo "Child Foo!", PHP_EOL; 
  73.  
  74.     } 
  75.  
  76.  
  77.     
  78.  
  79. $base = Child::getSelf(); 
  80.  
  81. $child = Child::getInstance(); 
  82.  
  83.     
  84.  
  85. $child->selfFoo(); 
  86.  
  87. $child->staticFoo(); 
  88.  
  89. $child->thisFoo(); 

程序输出结果如下:

  1. class Base { 
  2.  
  3.     public function __construct() { 
  4.  
  5.         echo "Base constructor!", PHP_EOL; 
  6.  
  7.     } 
  8.  
  9.     
  10.  
  11.     public static function getSelf() { 
  12.  
  13.         return new self(); 
  14.  
  15.     } 
  16.  
  17.     
  18.  
  19.     public static function getInstance() { 
  20.  
  21.         return new static(); 
  22.  
  23.     } 
  24.  
  25.     
  26.  
  27.     public function selfFoo() { 
  28.  
  29.         return self::foo(); 
  30.  
  31.     } 
  32.  
  33.     
  34.  
  35.     public function staticFoo() { 
  36.  
  37.         return static::foo(); 
  38.  
  39.     } 
  40.  
  41.     
  42.  
  43.     public function thisFoo() { 
  44.  
  45.         return $this->foo(); 
  46.  
  47.     } 
  48.  
  49.     
  50.  
  51.     public function foo() { 
  52.  
  53.         echo  "Base Foo!", PHP_EOL; 
  54.  
  55.     } 
  56.  
  57.  
  58.     
  59.  
  60. class Child extends Base { 
  61.  
  62.     public function __construct() { 
  63.  
  64.         echo "Child constructor!", PHP_EOL; 
  65.  
  66.     } 
  67.  
  68.     
  69.  
  70.     public function foo() { 
  71.  
  72.         echo "Child Foo!", PHP_EOL; 
  73.  
  74.     } 
  75.  
  76.  
  77.     
  78.  
  79. $base = Child::getSelf(); 
  80.  
  81. $child = Child::getInstance(); 
  82.  
  83.     
  84.  
  85. $child->selfFoo(); 
  86.  
  87. $child->staticFoo(); 
  88.  
  89. $child->thisFoo(); 

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

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

this

self与this是被讨论最多,也是最容易引起误用的组合。

两者的主要区别如下:

this不能用在静态成员函数中,self可以;

对静态成员函数/变量的访问,建议 用self,不要用$this::或$this->的形式;

对非静态成员变量的访问,不能用self,只能用this;

this要在对象已经实例化的情况下使用,self没有此限制;

在非静态成员函数内使用,self抑制多态行为,引用当前类的函数;而this引用调用类的重写(override)函数(如果有的话)。

self的用途

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

详细说则是:

替代类名,引用当前类的静态成员变量和静态函数;

抑制多态行为,引用当前类的函数而非子类中覆盖的实现;

槽点

这几个关键字中,只有this要加$符号且必须加,强迫症表示很难受;

静态成员函数中不能通过$this->调用非静态成员函数,但是可以通过self::调用,且在调用函数中未使用$this->的情况下还能顺畅运行。此行为貌似在不同PHP版本中表现不同,在当前的7.3中ok;

在静态函数和非静态函数中输出self,猜猜结果是什么?都是string(4) "self",迷之输出;

return $this instanceof static::class;会有语法错误,但是以下两种写法就正常:

  1. $class = static::class
  2.  
  3. return $this instanceof $class
  4.  
  5. // 或者这样: 
  6.  
  7. return $this instanceof static

所以这是为什么啊?!

  1. $class = static::class
  2.  
  3. return $this instanceof $class
  4.  
  5. // 或者这样: 
  6.  
  7. return $this instanceof static

以上就是解析PHP的self关键字的详细内容。

Tags: self

分享到: