PHP:“;调用未定义的方法“..但方法是有定义的


PHP: "Call to undefined method" ... but method is defined?

我有一个类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';
    }
});

使用这种机制,您就有了一个模仿命名空间结构的文件夹结构。有了这种方法,您就有了一个良好的结构,并且不需要关心包含库和其他类。