PHP 裏的 self::class、static::class 和 trait 中操做靜態屬性

在 PHP 裏,在沒有繼承時候,你用self::class 和 static::class是同樣的,都是獲取當前類名。php

可是若是用到了繼承,而且這個方法寫在了父類裏,你想要分別獲取當前父類名和未知的子類名,就要按照下面的方法進行獲取。測試

在 PHP 類中,self指向的是當前方法存在的這個類,也就是父類。static指向的是最終那個子類。code

class P
{
    public static function getParent()
    {
        return self::class;
    }
 
    public static function getChild()
    {
        return static::class;
    }
}
 
class C extends P
{
 
}
 
echo C::getParent(), PHP_EOL;
echo C::getChild(), PHP_EOL;

運行結果:繼承

P
C

你覺得這樣就結束了?不不不……get

今天心血來潮,寫到一個功能,可能會用到trait的static,寫了代碼測試了下……代碼就不貼了,沒錯,不管是self仍是static,都沒法獲取當前trait名。it

可是這不是重點,重點是下面的。我想要在trait中操做static屬性,因爲不清楚到底會指向誰,因此寫了代碼進行測試。io

trait T
{
    public static $data1 = [];
    public static $data2 = [];
    public static $data3 = [];
 
    public function setData($name, $value)
    {
        static::$data1[$name] = $value;
        self::$data2[$name] = $value;
        T::$data3[$name] = $value;
    }
}
 
class A
{
    use T;
}
 
class B extends A
{
     
}
 
$a = new A;
$b = new B;
$a->setData('a', 1);
$b->setData('b', 2);
 
echo '[static]', PHP_EOL;
var_dump(T::$data1, A::$data1, B::$data1);
echo PHP_EOL, '[self]', PHP_EOL;
var_dump(T::$data2, A::$data2, B::$data2);
echo PHP_EOL, '[T]', PHP_EOL;
var_dump(T::$data3, A::$data3, B::$data3);

運行結果:function

[static]
array(0) {
}
array(2) {
  ["a"]=>
  int(1)
  ["b"]=>
  int(2)
}
array(2) {
  ["a"]=>
  int(1)
  ["b"]=>
  int(2)
}
 
[self]
array(0) {
}
array(2) {
  ["a"]=>
  int(1)
  ["b"]=>
  int(2)
}
array(2) {
  ["a"]=>
  int(1)
  ["b"]=>
  int(2)
}
 
[T]
array(2) {
  ["a"]=>
  int(1)
  ["b"]=>
  int(2)
}
array(0) {
}
array(0) {
}

在trait中操做static屬性,不管是self仍是static,指向的都是那個類的static屬性。而只有直接使用trait名::靜態屬性,能夠指向到當前trait中存儲靜態屬性。class

相關文章
相關標籤/搜索