PHP:“调用未定义的方法”......但方法是否已定义?

时间:2012-07-30 06:13:56

标签: php inheritance static abstract-class

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

有谁能解释这里发生了什么?

2 个答案:

答案 0 :(得分:7)

类名PDO是PHP的标准类,因此,如果您的解决方案包含您的PDO课程,那么您实际上并不包括您的课程。

答案 1 :(得分:0)

我建议使用自动加载功能。它会让你的生活变得更简单,你可以摆脱所有require次呼叫。

PHP manual中有一个实现,我发现它非常有用。它可能看起来像这样:

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();

然而,我记得有一个错误。通常我有一个'bootstrap'文件,这个自动加载器:

spl_autoload_register(function($class) {
    if(file_exists('./lib/' . str_replace('\\', '/', $class) . '.php')) {
        require_once './lib/' . str_replace('\\', '/', $class) . '.php';
    }
});

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