我有一个类 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;
}
}
最初我认为我不必instance()
在 PDO 类中包含该函数,因为它继承自 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 类的地方......)。
为了解决这个错误,我使用了一个创可贴解决方案,我将 Model 构造函数中的 require 替换为:
if (!class_exists($database)) {
require_once 'Databases/' . $database . '.php';
}
谁能解释这里发生了什么?