依赖注入简单实现

Gia*_*lli 4 php oop design-patterns

在阅读完这个问题之后,我想知道是否有人可以帮助我了解如何使用这些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;
    }
}
Run Code Online (Sandbox Code Playgroud)

会话类:

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

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

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

    function __construct($session)
    {
        $this->session=$session;
        ...
    }
}
Run Code Online (Sandbox Code Playgroud)

外部:

$connection=new mysqli(DB_HOST, DB_USER, DB_PASSWORD, DB_DATABASE);
$database=new DBClass($connection);
$session=new Session($database);
$user=new User($session);
Run Code Online (Sandbox Code Playgroud)

这是正确的方法吗?

Gor*_*don 9

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

通过使用具体的DBClass作为TypeHint而不是接口,您仍然有一些耦合.因此,当您想要使用不同的DBClass时,必须将其命名为DBClass.您可以通过具体类必须实现的接口进行编码来实现更松散的耦合.

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

Symfony的组件库有依赖注入容器具有优异的文档和引进的话题就如何进一步加强这与DI容器服务.容器也可用于限制实例.看看这个.