克隆如何创建单例的新版本?

时间:2015-12-27 06:12:47

标签: php oop object clone

我问了this问题并得到了一个很好的解决方案。我理解如何在没有new的情况下创建对象(我认为这是构造对象的唯一方法)。

我了解到,通过创建一个单例类,我们强制一个类只实例化一次。

// a simple singleton class.
class Singleton
{
    private static $object = null;
    private function __construct() {}
    public static function createObject()
    {
        if (self::$object == null) {
            self::$object = new Singleton();
        }
        return self::$object;
    }
}
//$obj = new Singleton(); Obviously an error

$obj1 = Singleton::createObject();
// object(Singleton)[1]

$obj2 = Singleton::createObject();
// object(Singleton)[1]
// $obj1 and $obj2 are same. Both have same id = 1

$obj3 = clone $obj1;
// object(Singleton)[2]
// $obj3 is a new  instantiate. id = 2

这是怎么回事? clone如何在这里工作?

2 个答案:

答案 0 :(得分:2)

  

我了解到,通过创建一个单例类,我们强制一个类只实例化一次。

是的,如果我们使用我们实际设计的方法来创建对象。尽管singleton的想法是在整个应用程序中共享一个对象的单个实例,但它仍然是对象的实例

因此,可以随意克隆对象的实例。没有什么能阻止它被克隆。什么阻止重复实例是使用createObject()方法。 (对于记录,最好将其命名为getObject()

所以按顺序:

$obj1 = Singleton::createObject();
// the method call creates the instance with new
// assigns it to the property of the singleton class
// then returns it

$obj2 = Singleton::createObject();
// the method returns the instance from the first method call
// as it is still defined in the static property
// BOTH $obj1 and $obj2 are the same instance of an object

$obj3 = clone $obj1;
// here, we use the clone keyword.
// this has nothing to do with our singleton, yet
// it simply clones an instance of an object. 
// $obj1 is an instance of an object. It is therefore cloned.

现在,如果你真的希望clone能够返回相同的实例,你可以像MaxP所提到的那样在__clone()类中定义Singleton魔术方法。

function __clone()
{
    return self::createObject();
}

哪个会返回正确的实例,就像对createObject()的任何其他调用一样。

答案 1 :(得分:0)

如果您想阻止克隆单例,请添加私有方法__clone private function __clone() {}