依赖注入简单实现

时间:2010-01-08 20:48:01

标签: php oop design-patterns

阅读this问题后,我想知道是否有人可以帮助我理解如何使用这些PHP类正确实现依赖注入:

class DBClass
{
    private $mMysqli;
    function  __construct(mysqli $database)
    {
        $this->mMysqli=$database;
    }
    function __destruct()
    {
        $this->mMysqli->close();
    }

    public function listUsers()
    {
        $query='SELECT * FROM Utente;';
        $resultset=$this->mMysqli->query($query);
        while($row = $resultset->fetch_array(MYSQLI_ASSOC)) {
            echo $row['username'];
            echo $row['pwd'];
            echo "<br />\n";
        }

    }

    public function runQuery($query)
    {
        return $resultset=$this->mMysqli->query($query);
    }

    public function getConnection()
    {
        return $this->mMysqli;
    }
}

会话班:

class Session
{
    private $_session;
    public $maxTime;
    private $database;

    public function __construct(DBClass $database)
    {
        $this->database=$database;
        $this->maxTime['access'] = time();
        $this->maxTime['gc'] = get_cfg_var('session.gc_maxlifetime');
        session_set_save_handler(array($this,'_open'),
                array($this,'_close'),
                array($this,'_read'),
                array($this,'_write'),
                array($this,'_destroy'),
                array($this,'_clean')
                );
        register_shutdown_function('session_write_close');
        session_start();
        ...
     }
}   

用户类(有关当前登录用户的详细信息):

class User
{
    private $username;
    private $role;
    private $session;

    function __construct($session)
    {
        $this->session=$session;
        ...
    }
}

外部:

$connection=new mysqli(DB_HOST, DB_USER, DB_PASSWORD, DB_DATABASE);
$database=new DBClass($connection);
$session=new Session($database);
$user=new User($session);

这是正确的方法吗?

1 个答案:

答案 0 :(得分:9)

是。您现在通过构造函数注入依赖项,这提供了较少的耦合。您现在可以更轻松地更换这些依赖项,例如当UnitTesting你可以用模拟替换它们。

通过使用具体的DBClass作为TypeHint而不是interface,您仍然有一些耦合。因此,当您想要使用不同的DBClass时,必须将其命名为DBClass。您可以通过coding against an interface实现更具松散性的耦合,而具体类必须实现它。

要仅创建类的单个实例(如注释中所述),您可以使用Singleton(如Kevin Peno建议的)或Factory来创建并跟踪实例是否已创建。或者使用DI服务容器,它类似于工厂,但不是一回事。它为您创建和管理对象。

Symfony Components library has a Dependency Injection Container提供了优秀的文档和关于如何使用Service Containers进一步增强DI的主题的介绍。容器也可用于限制实例。看看吧。