如何使用PDO与依赖注入?

时间:2017-09-13 21:39:06

标签: php dependency-injection dependencies constructor-injection

我很难理解如何使用依赖注入。我在这里阅读了很多问题/答案,但是我无法用我正在使用的代码进行描述。

Model.php

abstract class Model {

    protected static function getDB() {
        static $db = null;

        if ($db === null) {
            $db = new PDO('mysql:host=host;dbname=dbname;charset=utf8', 'dbuser', 'password');
        }

        return $db;
    }
}

model.php只包含该功能,我想远离设置并将其称为静态

user.php的

class User extends Model {

    /*
    * Selects all of the user information
    */
    public function getUser($id){

        $db = static::getDB();

        $sth = $db->prepare('SELECT * FROM user WHERE id = :id');
        $sth->bindValue(':id', $id, PDO::PARAM_INT);
        $sth->execute();

        return $sth->fetch();
    }

    /*
    * Selects all of the user posts
    */
    public function getUserPosts($id){
        $db = static::getDB();

        $sth = $db->prepare('SELECT * FROM user_posts WHERE user_id = :id');
        $sth->bindValue(':id', $id, PDO::PARAM_INT);
        $sth->execute();

        return $sth->fetch();
    }
}

在user.php中我扩展了模型类,但我在每个函数中都设置了$db = static::getDB();

我知道依赖注入几乎只是将方法/变量传递给一个对象,但我甚至不确定我是否正确这样做。

更新了更多想法:

我认为创建私有变量会更好,而在构造函数中我们只需调用getDB()

class User extends Model {

    protected $db;

    public function __construct(){
        $this->db = getDB();
    }

    /*
    * example usage
    */
    public function getUser($id){
        $sth = $this->db->prepare('SELECT * FROM user WHERE id = :id');
        $sth->bindValue(':id', $id, PDO::PARAM_INT);
        $sth->execute();

        return $sth->fetch();
    }
}

但是它仍然算作依赖注入,因为我没有直接在函数构造函数中调用类吗?

第二次更新: 在阅读了多个指南之后,这个page让我感觉更有意义,这就是我想出来的。

model.php

abstract class Model {
    protected $db = null;

    public function __construct(){
        if($this->db === null){
            try {
                $this->db = new PDO('mysql:host=' . Config::DB_HOST . ';dbname=' . Config::DB_NAME . '; charset=utf8', Config::DB_USER, Config::DB_PASSWORD);
                $this->db->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
            } catch (PDOException $e) {
                echo 'Connection failed: ' . $e->getMessage();
            }
        }
        return $this->db;
    }
}

user.php的

class User extends Model {

    protected $db;

    public function __construct(Model $db){
        $this->db = $db;
    }

    /*
    * example usage
    */
    public function getUser($id){
        $sth = $this->db->prepare('SELECT * FROM user WHERE id = :id');
        $sth->bindValue(':id', $id, PDO::PARAM_INT);
        $sth->execute();

        return $sth->fetch();
    }
}

看起来如何?

1 个答案:

答案 0 :(得分:3)

我认为您没有使用依赖注入,因为您实际上并未向模型中提供任何依赖项,而是在构造函数上生成它们。

为了提供依赖关系,你应该将它作为参数传递给构造函数:

public function __construct($db){
    $this->db = $db;
}

通过这种方式,您可以将连接的创建与类分离,并且您可以使用依赖注入的好处,例如传递Mock对象进行测试而不是实际的事物。