欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

php static延迟静态绑定

程序员文章站 2022-12-01 18:07:21
如果你是一个懒惰的程序员,你看到以下代码可能会恼火     abstract  class U{      }...
如果你是一个懒惰的程序员,你看到以下代码可能会恼火

 

 

abstract  class U{

    

}

class u1 extends U{

   public static function create(){

     return new u1();

  }

  

  

}

 

class u2 extends U{

   public static function create(){

     return new u2();

   }

}

 

这段代码正常工作是没问题,但大量重复的代码会很烦人

 

我不想在每个子类中添加create方法,如果把create方法放在超类U中,代码可能是

 

 

abstract class U{

  public static function create(){

   return new self();

  }

}

class u1 extends U{

  function a(){}

}

 

class u2 extends U{

 

}

u1::create();

 

看起来很优雅整洁,现在我们把常见代码放在一个位置,并用self作为对该类的引用。但这里我们对self做了一个假设。

 

实际上,self对该类所起的作用与$this对对象所起的作用并不完全相同。self指的不是调用上下文,他指的是解析上下文,因此如果运行上面的列子,将会得到

 

Fatal error: Cannot instantiate abstract class U in D:\wamp\www\test\oop\static.php on line 21

 

因此self被解析为定义create的U,而不是解析为调用self的u1类。

 

php5.3之前,在这方面都有严格的限制,产生过很多笨拙的解决方案,php5.3引入了延迟静态绑定  及使用 关键字  static

 

static类似self,但它指的是被调用的类而不是包含类。

 

在以下例子中u1::create将生成u1对象,而不是实例化U对象

 

 

abstract class U{

  public static function create(){

   return new static();

  }

}

class u1 extends U{

}

 

class u2 extends U{

 

}

u1::create();

 

static不仅可以用于实例化,和self,parent一样还可以作为静态方法的调用标识符,甚至是从非静态上下文中调用

 

 

 

 

abstract class U{

 private $group;

  public function __construct(){

      $this->group=static::getGroup();

  }

  public static function create(){

   return new static();

  }

  static function getGroup(){

     return 'default';

  }

}

class u1 extends U{

}

 

class u2 extends U{

 static function getGroup(){

   return 'u2';

 }

}

class u3 extends u2{

 

}

print_r(u1::create());

echo '<br/>';

print_r(u3::create());

 

u1 Object ( [group:U:private] => default ) 

u3 Object ( [group:U:private] => u2 )