gpt4 book ai didi

php - 什么是与 PHP 框架的依赖注入(inject)相结合的良好 DAO 模式?

转载 作者:可可西里 更新时间:2023-10-31 23:32:23 25 4
gpt4 key购买 nike

我正在开发一个基于依赖注入(inject)的 PHP 框架。我的数据对象是可注入(inject)组件,就像其他任何组件一样。

我有一个抽象的 DAO 类,每个模型都应该扩展它,它有:

  • 基本的crud方法
  • 对 DI 容器的引用,用于实例化对象

简而言之,事情是这样的

abstract class AbstractDao {
protected $fields;
protected $container; // This is the (injected) DI container, used to create instances.
protected $driver; // The injected database driver (i.e. PDO)
public function insert() {
// Insert implementation
// Insert current instance.
}
public function fetch($id) {
// Fetch implementation
// Fetches a row and sets fields on current instance
}
public function fetchAll() {
// Performs a select * query on database driver
// Iterates through results, and creates an instance
// for each result using $container, like this:
foreach ($results as $row) {
// I can't just make $instances[] = new Something(), or all the
// dependency injection thing would mess up.
$instances[] = $this->container->get('someDao');
}
return $instances;
}
// Other methods.
}

class Book extends AbstractDao {
protected $fields = array('field', 'definition', 'goes', 'here',);
// No special behaviour is needed, so we can keep default
// abstract implementation without overriding.
}

我的问题:每个数据对象实现(一本书、一个人、一个用户等)都必须扩展我的 AbstractDao 对象,因此它将承载 $driver 和 $container 的重量。此外,由于 $fields 属性是在实例级别定义的,每个数据对象都有自己的属性,增加了更多的开销。

我担心在处理大数据集时,就性能而言,此解决方案可能会导致成本高昂。我知道对象只会被引用,而不是被克隆,但开销可能会非常高。

我想到的几个解决方案是

  • 使用静态方法实现,可以减少开销子类
  • 不要让我的 Daos 扩展上面提到的 AbstractDao,那应该成为一种 DaoProvider。在这种情况下,对于每个方法,我应该传入实例(我不太喜欢的东西)

我不太喜欢这些解决方案...首先我不喜欢使用静态的东西,因为它们与注入(inject)的整个想法有点冲突。其次,我不喜欢删除 Dao 子类化模式的想法。

任何好主意将不胜感激,谢谢。

=== 编辑 ===

我想到了另一件事。在第二种方法(“dao 提供者”)中我不喜欢的是提供者必须对 Dao 字段执行操作(设置值、设置状态、设置 isDirty 等),因此必须从外部访问字段.使用子类化方法,可以使那些 protected 或私有(private)。

===/EDIT ===

最佳答案

我建议您创建一个 DAO 接口(interface)来声明 DAO 实现必须定义的行为。现在,在每个具体的 DAO 实现中,您可以定义您的 $driver$container$fields 实例字段。

之后,您可能想要创建一个 AbstractModel 类,每个具体模型都应该扩展该类,这样您的 AbstractModel 和具体模型都将“与数据访问无关”。 AbstractModel 类最终看起来像这样:

/*
* an AbstractModel
*/
abstract class AbstractModel {

protected $daoImpl;

function __construct(DAOInterface $daoImpl) {
$this->daoImpl = $daoImpl;
}

//some other functions that are common to concrete models
}

/*
* a concrete model
*/
class Model extends AbstractModel {

function findAll($params) {
//You can use the $daoImpl of AbstractModel to perform a CRUD operation
$this->daoImpl->findAll($params);
}

}

现在,无论何时实例化一个具体模型,您都会将 DAO 实现注入(inject)到模型类中。

//inject a DAOInterface implementation into Model
$model = new Model(new DAOImpl());
$model->findAll($params);

这里的优点是您可以在测试期间 stub 不同的 DAO 实现,也许这就是 DI 容器派上用场的地方。有一个 similar code sample我在几天前创建我的 DI 容器时创建了。

顺便说一句,我不认为需要将 $container 对象放入您的 AbstractDAO 中,为什么不传入调用容器属性时返回的对象。这样你就可以使用 type hinting强制对象参数为特定类型,并在传入错误对象时鼓励快速失败机制,您可能还会发现创建一个 Config 类来处理您的 $driver 详细信息是有益的这样用户就可以自由配置他们想要用于数据库的驱动程序。

关于php - 什么是与 PHP 框架的依赖注入(inject)相结合的良好 DAO 模式?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15193891/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com