gpt4 book ai didi

php - 扩展抽象单例类

转载 作者:IT王子 更新时间:2023-10-28 23:56:32 28 4
gpt4 key购买 nike

如果您有一个创建某种新对象的工厂类,并且该 factroy 类是单例,如下所示:

class Database_Factory extends Base_Factory {
private static $factory;
private $objects = array();

public function __get($profile) {
// check for object and return it if it's created before
}

public static function getInstance(){
if (!self::$factory)
self::$factory = new self();
return self::$factory;
}
}

只要某个对象需要它自己的工厂,相同的代码就会重复出现。所以我决定把这个工厂类抽象化,只为每个工厂实现特定的例程。但是 PHP 不允许实例化抽象类。

abstract class Base_Factory {
public static function getInstance(){
if (!self::$factory)
self::$factory = new self();
return self::$factory;
}
}

fatal error :无法实例化抽象类 Base_Factory

你会怎么做?

最佳答案

在 PHP 方法中,self 总是指定义方法的类。从 5.3.0 版本开始,PHP 支持“后期静态绑定(bind)”,您可以使用 static 关键字访问覆盖的静态方法,以及函数 get_call_class() 来访问在静态上下文中获取派生类的名称。

但是,您的设计有一个重大缺陷:在 Base_Factory 中定义的静态属性 $factory 在所有派生类之间共享。因此,第一次创建单例并将其存储在此属性中时,所有其他对 getInstance() 的调用都将返回相同的对象,无论使用什么派生类。

您可以使用静态字典将类名映射到单例对象:

abstract class Base_Factory {
private static $_instances = array();
public static function getInstance() {
$class = get_called_class();
if (!isset(self::$_instances[$class])) {
self::$_instances[$class] = new $class();
}
return self::$_instances[$class];
}
}

哦,还有一件事:您正在寻找一种可能性来重用单例对象的代码,这可能暗示您过度使用单例设计模式!问问自己,您计划作为单例实现的类是否真的是单例,以及是否没有您可能希望拥有特定类的多个实例的用例。

通常最好只使用一个代表当前“应用程序上下文”的单例,它为与此上下文相关的单例对象提供访问器。

关于php - 扩展抽象单例类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1818765/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com