DDC-952 - Implemented first approach for batching eager loads of ToOne associations.
This commit is contained in:
parent
e0b835178b
commit
32df9451fd
4 changed files with 98 additions and 9 deletions
|
@ -59,6 +59,7 @@ class ObjectHydrator extends AbstractHydrator
|
||||||
$this->_resultPointers =
|
$this->_resultPointers =
|
||||||
$this->_idTemplate = array();
|
$this->_idTemplate = array();
|
||||||
$this->_resultCounter = 0;
|
$this->_resultCounter = 0;
|
||||||
|
$this->_hints['deferEagerLoad'] = true;
|
||||||
|
|
||||||
foreach ($this->_rsm->aliasMap as $dqlAlias => $className) {
|
foreach ($this->_rsm->aliasMap as $dqlAlias => $className) {
|
||||||
$this->_identifierMap[$dqlAlias] = array();
|
$this->_identifierMap[$dqlAlias] = array();
|
||||||
|
@ -132,6 +133,8 @@ class ObjectHydrator extends AbstractHydrator
|
||||||
$coll->takeSnapshot();
|
$coll->takeSnapshot();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$this->_em->getUnitOfWork()->triggerEagerLoads();
|
||||||
|
|
||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -560,7 +560,10 @@ class BasicEntityPersister
|
||||||
$result = $stmt->fetch(PDO::FETCH_ASSOC);
|
$result = $stmt->fetch(PDO::FETCH_ASSOC);
|
||||||
$stmt->closeCursor();
|
$stmt->closeCursor();
|
||||||
|
|
||||||
return $this->_createEntity($result, $entity, $hints);
|
$hints['deferEagerLoad'] = true;
|
||||||
|
$entity = $this->_createEntity($result, $entity, $hints);
|
||||||
|
$this->_em->getUnitOfWork()->triggerEagerLoads();
|
||||||
|
return $entity;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -577,6 +580,10 @@ class BasicEntityPersister
|
||||||
*/
|
*/
|
||||||
public function loadOneToOneEntity(array $assoc, $sourceEntity, $targetEntity, array $identifier = array())
|
public function loadOneToOneEntity(array $assoc, $sourceEntity, $targetEntity, array $identifier = array())
|
||||||
{
|
{
|
||||||
|
if ($foundEntity = $this->_em->getUnitOfWork()->tryGetById($identifier, $assoc['targetEntity'])) {
|
||||||
|
return $foundEntity;
|
||||||
|
}
|
||||||
|
|
||||||
$targetClass = $this->_em->getClassMetadata($assoc['targetEntity']);
|
$targetClass = $this->_em->getClassMetadata($assoc['targetEntity']);
|
||||||
|
|
||||||
if ($assoc['isOwningSide']) {
|
if ($assoc['isOwningSide']) {
|
||||||
|
@ -739,10 +746,13 @@ class BasicEntityPersister
|
||||||
$result = $stmt->fetchAll(PDO::FETCH_ASSOC);
|
$result = $stmt->fetchAll(PDO::FETCH_ASSOC);
|
||||||
$stmt->closeCursor();
|
$stmt->closeCursor();
|
||||||
|
|
||||||
|
$hints = array('deferEagerLoads' => true);
|
||||||
foreach ($result as $row) {
|
foreach ($result as $row) {
|
||||||
$entities[] = $this->_createEntity($row);
|
$entities[] = $this->_createEntity($row, null, $hints);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$this->_em->getUnitOfWork()->triggerEagerLoads();
|
||||||
|
|
||||||
return $entities;
|
return $entities;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -870,7 +880,14 @@ class BasicEntityPersister
|
||||||
|
|
||||||
$sql = $this->_getSelectEntitiesSQL($criteria, $assoc, 0, $limit, $offset);
|
$sql = $this->_getSelectEntitiesSQL($criteria, $assoc, 0, $limit, $offset);
|
||||||
list($params, $types) = $this->expandParameters($criteria);
|
list($params, $types) = $this->expandParameters($criteria);
|
||||||
return $this->_conn->executeQuery($sql, $params, $types);
|
$stmt = $this->_conn->executeQuery($sql, $params, $types);
|
||||||
|
$hints = array('deferEagerLoads' => true);
|
||||||
|
while ($result = $stmt->fetch(PDO::FETCH_ASSOC)) {
|
||||||
|
$coll->hydrateAdd($this->_createEntity($result, null, $hints));
|
||||||
|
}
|
||||||
|
$stmt->closeCursor();
|
||||||
|
|
||||||
|
$this->_em->getUnitOfWork()->triggerEagerLoads();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1330,7 +1347,16 @@ class BasicEntityPersister
|
||||||
|
|
||||||
$sql = $this->_getSelectEntitiesSQL($criteria, $assoc, 0, $limit, $offset);
|
$sql = $this->_getSelectEntitiesSQL($criteria, $assoc, 0, $limit, $offset);
|
||||||
list($params, $types) = $this->expandParameters($criteria);
|
list($params, $types) = $this->expandParameters($criteria);
|
||||||
return $this->_conn->executeQuery($sql, $params, $types);
|
|
||||||
|
$stmt = $this->_conn->executeQuery($sql, $params, $types);
|
||||||
|
$hints = array('deferEagerLoads' => true);
|
||||||
|
while ($result = $stmt->fetch(PDO::FETCH_ASSOC)) {
|
||||||
|
$coll->hydrateAdd($this->_createEntity($result, null, $hints));
|
||||||
|
}
|
||||||
|
$stmt->closeCursor();
|
||||||
|
|
||||||
|
$this->_em->getUnitOfWork()->triggerEagerLoads();
|
||||||
|
return $stmt;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -1928,10 +1928,18 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
$newValue = $this->getEntityPersister($assoc['targetEntity'])
|
$newValue = $this->getEntityPersister($assoc['targetEntity'])
|
||||||
->loadOneToOneEntity($assoc, $entity, null, $associatedId);
|
->loadOneToOneEntity($assoc, $entity, null, $associatedId);
|
||||||
} else {
|
} else {
|
||||||
if ($assoc['fetch'] == ClassMetadata::FETCH_EAGER) {
|
if ($assoc['fetch'] == ClassMetadata::FETCH_EAGER && isset($hints['deferEagerLoad'])) {
|
||||||
// TODO: Maybe it could be optimized to do an eager fetch with a JOIN inside
|
if (!isset($this->eagerLoadingEntities[$assoc['targetEntity']])) {
|
||||||
// the persister instead of this rather unperformant approach.
|
$this->eagerLoadingEntities[$assoc['targetEntity']] = array();
|
||||||
$newValue = $this->em->find($assoc['targetEntity'], $associatedId);
|
}
|
||||||
|
|
||||||
|
// TODO: Is there a faster approach?
|
||||||
|
$this->eagerLoadingEntities[$assoc['targetEntity']] = array_merge_recursive(
|
||||||
|
$this->eagerLoadingEntities[$assoc['targetEntity']],
|
||||||
|
array_map(function($id) {
|
||||||
|
return array($id);
|
||||||
|
}, $associatedId)
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
$newValue = $this->em->getProxyFactory()->getProxy($assoc['targetEntity'], $associatedId);
|
$newValue = $this->em->getProxyFactory()->getProxy($assoc['targetEntity'], $associatedId);
|
||||||
}
|
}
|
||||||
|
@ -1983,6 +1991,26 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
<<<<<<< HEAD
|
||||||
|
=======
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function triggerEagerLoads()
|
||||||
|
{
|
||||||
|
if (!$this->eagerLoadingEntities) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$eagerLoadingEntities = $this->eagerLoadingEntities;
|
||||||
|
$this->eagerLoadingEntities = array();
|
||||||
|
|
||||||
|
foreach ($eagerLoadingEntities AS $entityName => $ids) {
|
||||||
|
$this->getEntityPersister($entityName)->loadAll($ids);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
>>>>>>> DDC-952 - Implemented first approach for batching eager loads of ToOne associations.
|
||||||
* Initializes (loads) an uninitialized persistent collection of an entity.
|
* Initializes (loads) an uninitialized persistent collection of an entity.
|
||||||
*
|
*
|
||||||
* @param PeristentCollection $collection The collection to initialize.
|
* @param PeristentCollection $collection The collection to initialize.
|
||||||
|
|
|
@ -21,6 +21,10 @@ class DDC633Test extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @group DDC-633
|
||||||
|
* @group DDC-952
|
||||||
|
*/
|
||||||
public function testOneToOneEager()
|
public function testOneToOneEager()
|
||||||
{
|
{
|
||||||
$app = new DDC633Appointment();
|
$app = new DDC633Appointment();
|
||||||
|
@ -35,7 +39,35 @@ class DDC633Test extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
|
|
||||||
$eagerAppointment = $this->_em->find(__NAMESPACE__ . '\DDC633Appointment', $app->id);
|
$eagerAppointment = $this->_em->find(__NAMESPACE__ . '\DDC633Appointment', $app->id);
|
||||||
|
|
||||||
$this->assertNotType('Doctrine\ORM\Proxy\Proxy', $eagerAppointment->patient);
|
// Eager loading still produces proxies
|
||||||
|
$this->assertType('Doctrine\ORM\Proxy\Proxy', $eagerAppointment->patient);
|
||||||
|
$this->assertTrue($eagerAppointment->patient->__isInitialized__, "Proxy should already be initialized due to eager loading!");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @group DDC-633
|
||||||
|
* @group DDC-952
|
||||||
|
*/
|
||||||
|
public function testDQLDeferredEagerLoad()
|
||||||
|
{
|
||||||
|
for ($i = 0; $i < 10; $i++) {
|
||||||
|
$app = new DDC633Appointment();
|
||||||
|
$pat = new DDC633Patient();
|
||||||
|
$app->patient = $pat;
|
||||||
|
$pat->appointment = $app;
|
||||||
|
|
||||||
|
$this->_em->persist($app);
|
||||||
|
$this->_em->persist($pat);
|
||||||
|
}
|
||||||
|
$this->_em->flush();
|
||||||
|
$this->_em->clear();
|
||||||
|
|
||||||
|
$appointments = $this->_em->createQuery("SELECT a FROM " . __NAMESPACE__ . "\DDC633Appointment a")->getResult();
|
||||||
|
|
||||||
|
foreach ($appointments AS $eagerAppointment) {
|
||||||
|
$this->assertType('Doctrine\ORM\Proxy\Proxy', $eagerAppointment->patient);
|
||||||
|
$this->assertTrue($eagerAppointment->patient->__isInitialized__, "Proxy should already be initialized due to eager loading!");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue