阅读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);
这是正确的方法吗?
答案 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的主题的介绍。容器也可用于限制实例。看看吧。