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

Log*_*man 3 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;
    }
}

?>
Run Code Online (Sandbox Code Playgroud)

其他数据库扩展了这个类并实现了抽象函数,这样我就可以更改我使用的数据库,同时确保我的应用程序仍然有效,只要它使用这个抽象层中的函数(我已经省略了上面的函数定义)码).

我在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;
    }
}
Run Code Online (Sandbox Code Playgroud)

最初我以为我不必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
    }
}

?>
Run Code Online (Sandbox Code Playgroud)

现在这段代码给了我一个错误,我无法重新声明PDO类.我在网上搜索,发现问题通常与使用include()而不是require_once().我到处检查了我的Autoloader,没有看到任何include().即使require_once()在上面的模型中,它仍然给我错误(这是我唯一需要PDO类的地方......).

为了解决这个错误,我正在使用一个创可贴解决方案,我在模型构造函数中替换require:

if (!class_exists($database)) {
    require_once 'Databases/' . $database . '.php';
}
Run Code Online (Sandbox Code Playgroud)

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

enr*_*cog 7

类名PDO是PHP的标准类,因此,如果您的解决方案包含您的PDO类,您实际上从不包括您的类.

  • Gaaaaaahhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhhh谢谢你 (2认同)
  • 这会产生不同的错误."无法重新声明类PDO" - 完全有可能拥有一个早于PDO的PHP版本,或者特别是在编译时禁用了PDO. (2认同)