我有一个类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()
有关。我检查了所有地方,我的自动加载器没有看到任何include()
。即使上面的模型中有require_once()
,它仍然会给我错误(这是我唯一需要PDO类的地方…)
为了修复这个错误,我使用了一个创可贴解决方案,将Model构造函数中的require替换为:
if (!class_exists($database)) {
require_once 'Databases/' . $database . '.php';
}
有人能解释一下这里发生了什么吗?
类名PDO
是PHP的标准类,因此使用包含PDO
类的解决方案,实际上永远不会包含您的类。
我建议使用自动加载。这会让你的生活更简单,你可以摆脱所有那些require
电话。
PHP手册中有一个实现,我发现它非常有用。它可能看起来像这样:
function my_autoloader($class) {
include 'classes/' . $class . '.class.php';
}
spl_autoload_register('my_autoloader');
// Or, using an anonymous function as of PHP 5.3.0
spl_autoload_register(function ($class) {
include 'classes/' . $class . '.class.php';
});
如果您使用的是PHP 5.3,也可以考虑使用名称空间。事实上,拥有一个良好的结构和一个默认实现的简单自动加载器更容易:
spl_autoload_extensions(".php"); // comma-separated list
spl_autoload_register();
然而,我记得那里有一个bug。通常我有一个"引导"文件,使用这个自动加载器:
spl_autoload_register(function($class) {
if(file_exists('./lib/' . str_replace('''', '/', $class) . '.php')) {
require_once './lib/' . str_replace('''', '/', $class) . '.php';
}
});
使用这种机制,您就有了一个模仿命名空间结构的文件夹结构。有了这种方法,您就有了一个良好的结构,并且不需要关心包含库和其他类。