【发布时间】:2012-07-27 19:24:54
【问题描述】:
我有一个类 Database.php,它是一个抽象的 Singleton 类:
<?php
abstract class Database
{
protected static $_instance;
...
public static function instance()
{
$class = get_called_class();
if (!self::$_instance) {
self::$_instance = new $class();
}
return self::$_instance;
}
}
?>
其他数据库扩展了这个类并实现了抽象函数,这样我就可以改变我使用的数据库,同时确保我的应用程序仍然可以工作,只要它使用这个抽象层中的函数(我省略了函数定义从上面的代码)。
我在 PDO.php 中有一个 PDO 类。它扩展了数据库,还包括与上述相同的instance() 函数,但它没有自己的$_instance 变量。
class PDO extends Database
{
//...
public static function instance()
{
$class = get_called_class();
if (!self::$_instance) {
self::$_instance = new $class();
}
return self::$_instance;
}
}
最初我认为我不必在 PDO 类中包含 instance() 函数,因为它继承自 Database。但我包含它是因为我收到了这个错误:
Fatal error: Call to undefined method PDO::instance()
问题是我仍然收到此错误,即使使用上面的代码也是如此。我不知道这是否相关,但我遇到了另一个奇怪的错误。我要使用的数据库类型(在本例中为 PDO)存储在可通过App::setting('DB') 访问的变量中。我的 MVC 框架中的 Model 基类会加载相应的数据库类并将其存储在 $this->_db 中。这是模型的代码:
<?php
require_once 'Databases/Database.php';
class Model
{
protected $_db;
public function __construct()
{
$database = App::setting('DB'); // 'PDO'
require_once 'Databases/' . $database . '.php';
$this->_db = $database::instance(); // this is what triggers the error
}
}
?>
现在这段代码给了我一个错误,我无法重新声明 PDO 类。我在网上搜索,发现问题通常与使用include()而不是require_once()有关。我到处检查,我的 Autoloader 没有看到任何 include()。即使在上面的模型中使用require_once(),它仍然给我错误(这是我唯一需要 PDO 类的地方......)。
为了解决这个错误,我使用了一种创可贴解决方案,我将模型构造函数中的 require 替换为:
if (!class_exists($database)) {
require_once 'Databases/' . $database . '.php';
}
谁能解释这里发生了什么?
【问题讨论】:
-
顺便说一下Logan,以防万一您不知道,您也可以扩展内置类。看到
class Database extends PDO并不少见。尽管您应该远离单例并尽可能使用依赖注入。 -
我不想扩展 PDO。我使用的是单例,因为我只需要一个用于任何模型的数据库对象,这样如果我要加载两个模型,它们就不会连接到数据库两次。有没有更好的方法来实现这一目标?我不知道依赖注入是如何工作的,但我不是专家。很多 PHP 框架都使用单例,所以我就使用了。
标签: php inheritance static abstract-class