为什么要返回新静态? (PHP)

时间:2016-05-26 12:15:33

标签: php class oop static late-static-binding

为什么有些人会创建一个返回new static而不是将所有方法设为静态的方法?使用返回新静态的方法的原因是什么? 我不是在问静态和自我之间有什么区别,或者静态&自我意味着。例如,这是一个简单的类:

<?php

class Expression
{
    public static function make()
    {
        return new static;
    }


    public function find($value)
    {
        return '/' . $value .'/';
    }

    public function then($value)  
    {
        return $this->find($value);
    }

    public function hi($value)  
    {
        return "hi";
    }

}

如您所见,静态方法 make()会返回新静态。然后,有些人会调用其他方法:

$regex = Expression::make()->find('www');

它的目的是什么?我在这里看到我们不使用 new Expression 语法,如果这就是重点 - 那么为什么他们没有使所有方法成为例如静态?有什么区别,有什么理由让这个方法返回 new static

2 个答案:

答案 0 :(得分:22)

new static从当前类中实例化一个新对象,并使用后期静态绑定(如果该类是子类,则实例化子类,我希望你理解这一点。)

在类上返回一个新实例的static方法是替代构造函数。意思是,通常你的构造函数是public function __construct,通常需要一堆参数:

class Foo {
    public function __construct(BarInterface $bar, array $baz = []) { ... }
}

拥有替代构造函数允许您提供不同的默认值或方便快捷方式来实例化此类,而无需提供这些特定参数和/或能够提供替代构造函数将提供的不同参数转换为规范的:

class Foo {

    public function __construct(BarInterface $bar, array $baz = []) { ... }

    public static function fromBarString($bar) {
        return new static(new Bar($bar));
    }

    public static function getDefault() {
        return new static(new Bar('baz'), [42]);
    }

}

现在,即使您的规范构造函数需要一堆复杂的参数,您也可以创建一个类的默认实例,这对于大多数用途来说可能没问题,只需使用Foo::getDefault()

PHP中的典型示例是DateTimeDateTime::createFromFormat

在你的具体例子中,替代构造函数实际上并没有做任何事情,所以它相当多余,但我希望它是因为它是一个不完整的例子。如果确实有一个替代构造函数除了new static之外什么都不做,那么它可能仅仅意味着(new Foo)->上的便利语法,我觉得这是有问题的。

答案 1 :(得分:3)

get_called_class())。

class A {
    public static function get_self() {
        return new self();
    }

    public static function get_static() {
        return new static();
    }
}

class B extends A {}

echo get_class(B::get_self());  // A
echo get_class(B::get_static()); // B
echo get_class(A::get_self()); // A
echo get_class(A::get_static()); // A