gpt4 book ai didi

orm - Doctrine ORM Single Table Inheritance association problem (always Eager loading contractoring to documentation))

转载 作者:行者123 更新时间:2023-12-04 14:40:26 25 4
gpt4 key购买 nike

我有一个单表继承问题,我不确定我是否正确地解释了文档。

首先:我没有在这里详细复制我的代码/实体映射(甚至使用正确的语法),因为我认为可以更好地抽象地传达问题。 如果这无论如何都不能接受,请说出来,我将添加完整的代码 - 但它很长,我认为没有它也可以回答我的问题。

如果有帮助我会画画 一个 ER 图来尝试传达我正在尝试做的事情。 如果您阅读以下内容并认为“嘿,这应该行得通”——请告诉我,我会上传真实代码

第二: 我不在任何地方使用延迟加载。在访问我的实体之前,我确保通过预先编写 DQL 来加载我将要访问的每个相关实体 - 因此以下问题对我的应用程序来说是相当重要的)

设置:

我有一些实体 - 它们构成了我的应用程序的核心。

// @entity AnimalTrainingSchool
// oneToMany: trainingDepartment
fields: [name / location / year founded]
// @entity trainingDepartment
oneToMany: animalTrainer
oneToOne: building
fields: [capacity]
// @entity animalTrainer
fields: [name / age / qualification]

我经常在不同的上下文中访问它们 - 但我通常遍历级别并访问这些实体的属性和关系。

foreach ($animalTrainingSchool as $school){
echo $school->getName() . ' at ' . $school->getLocation();
echo 'These are the training departments for this school:';
foreach ($school->getTrainingDepartments as $trainingDepartment){
echo $trainingDepartment->getAnimalTypeTrained() . ' are trained in this department';
}
}

我通过形成我的 DQL 并执行它来确保所有这些都预先加载 - 以限制 SQL 查询的数量(一个)。这一切都很好(相当标准的东西)。

DQL Example : "SELECT ats, td, at FROM AnimalTrainingSchool ats JOIN AnimalTrainingSchool.trainingDepartments td JOIN td.animalTrainer at";

这设置了我的集合,意味着我可以遍历它而无需发出额外的查询

问题:

我已经在我的应用程序的其他地方映射了其他实体 - 与此非常相似(注意:我的总体问题与下面的问题非常相似,只有一个主要区别(见下文)

Doctrine 2 Inheritance Mapping with Association

// NB: new awards are issued each time they are awarded - so it is meant to be a oneToOne relationships - the awards are unique
// @entity award
{id information / table information / discriminator map / inheritance type (SINGLE_TABLE)}
fields: [medalMaterial / award_serial_number]
//departmentAward extends award
oneToOne: trainingDepartment
//trainerAward extends award
oneToOne: animalTrainer

然后我通过修改我的初始实体使关系成为双向的

// @entity trainingDepartment
oneToMany: animalTrainer
oneToOne: building
oneToOne: departmentAward
fields: [capacity]
// @entity animalTrainer
fields: [name / age / qualification]
oneToOne: trainerAward

发生了什么

现在,当我以与上述完全相同的方式访问我的原始实体时 - 他们会自动(急切地)加载相关实体以获得他们的奖励,尽管我没有告诉他们这样做。当我遍历一大堆 trainingDepartments/AnimalTrainers 和 Doctrine 正在为每个实体执行 SQL 语句时,这尤其糟糕。

对于 20 个部门,每个部门有 10 名培训师 - 这是 200 个额外的查询。

//Given the exact same DQL and foreach loop as above (note that at no stage am I accessing awards) - I get a ton of extra queries that look like this

"SELECT award.property1, award.property2, award.property3 FROM awardTable LEFT JOIN trainingDepartmentTable ON award.awardee_id = trainingDepartmentTable.id and award.discriminatorColumn IN ('departmentAward')";
// or...
"SELECT award.property1, award.property2, award.property3 FROM awardTable LEFT JOIN animalTrainerTable ON award.awardee_id = animalTrainerTable.id and award.discriminatorColumn IN ('trainerAward')";

生成的内容没有一个是不正确的 - 只是阅读了以下问题,在我看来我已经按照文档描述的方式设置了它(并且与@Matthieu相反;即 - 如果我相关我最初的 3 个实体到最低级别的实体而不是“奖励”基类,那么他们应该能够使用代理而不是尝试急切地加载实体。

Stackoverflow 问题与我所描述的相反

Doctrine 2 Inheritance Mapping with Association

相关理论文档

http://www.doctrine-project.org/docs/orm/2.0/en/reference/inheritance-mapping.html#performance-impact

There is a general performance consideration with Single Table Inheritance: If you use a STI entity as a many-to-one or one-to-one entity you should never use one of the classes at the upper levels of the inheritance hierachy as “targetEntity”, only those that have no subclasses. Otherwise Doctrine CANNOT create proxy instances of this entity and will ALWAYS load the entity eagerly.

在我看来,无论您是加入基本级别的实体还是子类实体 - Doctrine 都会急切地加载关联并且不会尝试使用代理。

再说一次:我可以发布真实代码 - 但考虑到问题的长度,我觉得最好不要发布。非常感谢任何输入。

最佳答案

oneToOne 关系的反面不能延迟加载。为了支持延迟加载,Doctrine 需要创建一个代理对象,但是代理对象需要有一个与之关联的标识符。在 oneToOne 关系的情况下,标识符仅在拥有方可用。所以必须急切加载反向关系。

如果可能,您应该尝试获取加入这些协会。 2.1 版将自动强制获取反向 oneToOne 关系的连接。

关于orm - Doctrine ORM Single Table Inheritance association problem (always Eager loading contractoring to documentation)),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6380097/

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