您尝试以与使用parent :: __ construct()相同的方式使用A_Factory :: create_A().但是,这是两个完全不同的调用.
父:: __构建体()
parent解析为A.当前对象$this是A的实例(因为B的每个实例由于继承而也是A的实例).在这种情况下,虽然运算符::已被使用($this保持不变),但调用不是静态调用.
以下代码有效:
(假设foo不是私有的)
class B extends A
{
function __construct()
{
parent::__construct(new X, new Y, new Z);
echo $this->foo;
}
}
这个也有效:
class B extends A
{
function __construct()
{
A::__construct(new X, new Y, new Z);
echo $this->foo;
}
}
厂
A_Factory :: createA()是一个静态调用,因为A_Factory不在B的继承树中.A_Factory也创建并返回A的新实例.所以一旦调用它,就会有两个不同的对象:$this仍然是未更改的B实例,您创建了一个不同的A实例,而不将其分配给任何变量.
一种可能的方法是将工厂方法移动到A本身.
这将有效:
class A
{
function __construct(X $x, Y $y, Z $z)
{
$this->foo = 'foo';
}
public static function create()
{
return new static (new X, new Y, new Z);
}
}
class B extends A
{
}
// Instantiating:
$a = A::create();
$b = B::create();
这将使用带有static关键字的late static binding. static解析为被调用类的类名,因此它在A :: create()中是A,在B :: create()中是B.
注意与self的区别,它总是解析为声明方法的类(在这种情况下它总是A)