From 9cd16ec56a44e0dc089e6de5174e69bdca6bf2b2 Mon Sep 17 00:00:00 2001 From: Guilherme Blanco <guilhermeblanco@gmail.com> Date: Mon, 28 Apr 2014 13:47:57 +0000 Subject: [PATCH] More optimizations around hydrators. Pretty much same performance, just better memory footprint (-0.25MB). --- .../Internal/Hydration/AbstractHydrator.php | 42 +++++++++---------- .../ORM/Internal/Hydration/ArrayHydrator.php | 9 ++-- .../ORM/Internal/Hydration/ObjectHydrator.php | 42 ++++++++----------- .../Hydration/SimpleObjectHydrator.php | 40 ++++++------------ 4 files changed, 52 insertions(+), 81 deletions(-) diff --git a/lib/Doctrine/ORM/Internal/Hydration/AbstractHydrator.php b/lib/Doctrine/ORM/Internal/Hydration/AbstractHydrator.php index f732bc099..bf43235be 100644 --- a/lib/Doctrine/ORM/Internal/Hydration/AbstractHydrator.php +++ b/lib/Doctrine/ORM/Internal/Hydration/AbstractHydrator.php @@ -263,13 +263,14 @@ abstract class AbstractHydrator continue; } + $fieldName = $cacheKeyInfo['fieldName']; + switch (true) { case (isset($cacheKeyInfo['isNewObjectParameter'])): - $fieldName = $cacheKeyInfo['fieldName']; - $argIndex = $cacheKeyInfo['argIndex']; - $objIndex = $cacheKeyInfo['objIndex']; - $type = $cacheKeyInfo['type']; - $value = $type->convertToPHPValue($value, $this->_platform); + $argIndex = $cacheKeyInfo['argIndex']; + $objIndex = $cacheKeyInfo['objIndex']; + $type = $cacheKeyInfo['type']; + $value = $type->convertToPHPValue($value, $this->_platform); $rowData['newObjects'][$objIndex]['class'] = $cacheKeyInfo['class']; $rowData['newObjects'][$objIndex]['args'][$argIndex] = $value; @@ -278,15 +279,15 @@ abstract class AbstractHydrator break; case (isset($cacheKeyInfo['isScalar'])): - $value = $cacheKeyInfo['type']->convertToPHPValue($value, $this->_platform); + $type = $cacheKeyInfo['type']; + $value = $type->convertToPHPValue($value, $this->_platform); - $rowData['scalars'][$cacheKeyInfo['fieldName']] = $value; + $rowData['scalars'][$fieldName] = $value; break; case (isset($cacheKeyInfo['isMetaColumn'])): - $dqlAlias = $cacheKeyInfo['dqlAlias']; - $fieldName = $cacheKeyInfo['fieldName']; - $type = $cacheKeyInfo['type']; + $dqlAlias = $cacheKeyInfo['dqlAlias']; + $type = $cacheKeyInfo['type']; // Avoid double setting or null assignment if (isset($rowData['data'][$dqlAlias][$fieldName]) || $value === null) { @@ -298,17 +299,14 @@ abstract class AbstractHydrator $nonemptyComponents[$dqlAlias] = true; } - if ($type) { - $value = $type->convertToPHPValue($value, $this->_platform); - } - - $rowData['data'][$dqlAlias][$fieldName] = $value; + $rowData['data'][$dqlAlias][$fieldName] = $type + ? $type->convertToPHPValue($value, $this->_platform) + : $value; break; default: - $dqlAlias = $cacheKeyInfo['dqlAlias']; - $fieldName = $cacheKeyInfo['fieldName']; - $type = $cacheKeyInfo['type']; + $dqlAlias = $cacheKeyInfo['dqlAlias']; + $type = $cacheKeyInfo['type']; // in an inheritance hierarchy the same field could be defined several times. // We overwrite this value so long we don't have a non-null value, that value we keep. @@ -375,11 +373,9 @@ abstract class AbstractHydrator $dqlAlias = $cacheKeyInfo['dqlAlias']; $type = $cacheKeyInfo['type']; - if ($type) { - $value = $type->convertToPHPValue($value, $this->_platform); - } - - $rowData[$dqlAlias . '_' . $fieldName] = $value; + $rowData[$dqlAlias . '_' . $fieldName] = $type + ? $type->convertToPHPValue($value, $this->_platform) + : $value; break; default: diff --git a/lib/Doctrine/ORM/Internal/Hydration/ArrayHydrator.php b/lib/Doctrine/ORM/Internal/Hydration/ArrayHydrator.php index 13ce8ff5b..12bda0581 100644 --- a/lib/Doctrine/ORM/Internal/Hydration/ArrayHydrator.php +++ b/lib/Doctrine/ORM/Internal/Hydration/ArrayHydrator.php @@ -67,11 +67,7 @@ class ArrayHydrator extends AbstractHydrator */ protected function prepare() { - $this->_isSimpleQuery = count($this->_rsm->aliasMap) <= 1; - $this->_identifierMap = array(); - $this->_resultPointers = array(); - $this->_idTemplate = array(); - $this->_resultCounter = 0; + $this->_isSimpleQuery = count($this->_rsm->aliasMap) <= 1; foreach ($this->_rsm->aliasMap as $dqlAlias => $className) { $this->_identifierMap[$dqlAlias] = array(); @@ -134,7 +130,8 @@ class ArrayHydrator extends AbstractHydrator } $relationAlias = $this->_rsm->relationMap[$dqlAlias]; - $relation = $this->getClassMetadata($this->_rsm->aliasMap[$parent])->associationMappings[$relationAlias]; + $parentClass = $this->_metadataCache[$this->_rsm->aliasMap[$parent]]; + $relation = $parentClass->associationMappings[$relationAlias]; // Check the type of the relation (many or single-valued) if ( ! ($relation['type'] & ClassMetadata::TO_ONE)) { diff --git a/lib/Doctrine/ORM/Internal/Hydration/ObjectHydrator.php b/lib/Doctrine/ORM/Internal/Hydration/ObjectHydrator.php index 044d85261..c3e4ca7c7 100644 --- a/lib/Doctrine/ORM/Internal/Hydration/ObjectHydrator.php +++ b/lib/Doctrine/ORM/Internal/Hydration/ObjectHydrator.php @@ -42,22 +42,22 @@ class ObjectHydrator extends AbstractHydrator /** * @var array */ - private $identifierMap; + private $identifierMap = array(); /** * @var array */ - private $resultPointers; + private $resultPointers = array(); /** * @var array */ - private $idTemplate; + private $idTemplate = array(); /** * @var integer */ - private $resultCounter; + private $resultCounter = 0; /** * @var array @@ -79,12 +79,6 @@ class ObjectHydrator extends AbstractHydrator */ protected function prepare() { - $this->identifierMap = - $this->resultPointers = - $this->idTemplate = array(); - - $this->resultCounter = 0; - if ( ! isset($this->_hints[UnitOfWork::HINT_DEFEREAGERLOAD])) { $this->_hints[UnitOfWork::HINT_DEFEREAGERLOAD] = true; } @@ -93,25 +87,23 @@ class ObjectHydrator extends AbstractHydrator $this->identifierMap[$dqlAlias] = array(); $this->idTemplate[$dqlAlias] = ''; - if ( ! isset($this->ce[$className])) { - $this->ce[$className] = $this->_em->getClassMetadata($className); - } - // Remember which associations are "fetch joined", so that we know where to inject // collection stubs or proxies and where not. if ( ! isset($this->_rsm->relationMap[$dqlAlias])) { continue; } - if ( ! isset($this->_rsm->aliasMap[$this->_rsm->parentAliasMap[$dqlAlias]])) { - throw HydrationException::parentObjectOfRelationNotFound($dqlAlias, $this->_rsm->parentAliasMap[$dqlAlias]); + $parent = $this->_rsm->parentAliasMap[$dqlAlias]; + + if ( ! isset($this->_rsm->aliasMap[$parent])) { + throw HydrationException::parentObjectOfRelationNotFound($dqlAlias, $parent); } - $sourceClassName = $this->_rsm->aliasMap[$this->_rsm->parentAliasMap[$dqlAlias]]; + $sourceClassName = $this->_rsm->aliasMap[$parent]; $sourceClass = $this->getClassMetadata($sourceClassName); $assoc = $sourceClass->associationMappings[$this->_rsm->relationMap[$dqlAlias]]; - $this->_hints['fetched'][$this->_rsm->parentAliasMap[$dqlAlias]][$assoc['fieldName']] = true; + $this->_hints['fetched'][$parent][$assoc['fieldName']] = true; if ($assoc['type'] === ClassMetadata::MANY_TO_MANY) { continue; @@ -126,7 +118,7 @@ class ObjectHydrator extends AbstractHydrator // handle fetch-joined owning side bi-directional one-to-one associations if ($assoc['inversedBy']) { - $class = $this->ce[$className]; + $class = $this->getClassMetadata($className); $inverseAssoc = $class->associationMappings[$assoc['inversedBy']]; if ( ! ($inverseAssoc['type'] & ClassMetadata::TO_ONE)) { @@ -198,7 +190,7 @@ class ObjectHydrator extends AbstractHydrator if ( ! $value instanceof PersistentCollection) { $value = new PersistentCollection( - $this->_em, $this->ce[$relation['targetEntity']], $value + $this->_em, $this->_metadataCache[$relation['targetEntity']], $value ); $value->setOwner($entity, $relation); @@ -255,7 +247,7 @@ class ObjectHydrator extends AbstractHydrator throw HydrationException::emptyDiscriminatorValue($dqlAlias); } - $discrMap = $this->ce[$className]->discriminatorMap; + $discrMap = $this->_metadataCache[$className]->discriminatorMap; if ( ! isset($discrMap[$data[$discrColumn]])) { throw HydrationException::invalidDiscriminatorValue($data[$discrColumn], array_keys($discrMap)); @@ -267,7 +259,7 @@ class ObjectHydrator extends AbstractHydrator } if (isset($this->_hints[Query::HINT_REFRESH_ENTITY]) && isset($this->rootAliases[$dqlAlias])) { - $this->registerManaged($this->ce[$className], $this->_hints[Query::HINT_REFRESH_ENTITY], $data); + $this->registerManaged($this->_metadataCache[$className], $this->_hints[Query::HINT_REFRESH_ENTITY], $data); } $this->_hints['fetchAlias'] = $dqlAlias; @@ -284,7 +276,7 @@ class ObjectHydrator extends AbstractHydrator private function getEntityFromIdentityMap($className, array $data) { // TODO: Abstract this code and UnitOfWork::createEntity() equivalent? - $class = $this->ce[$className]; + $class = $this->_metadataCache[$className]; /* @var $class ClassMetadata */ if ($class->isIdentifierComposite) { @@ -352,7 +344,7 @@ class ObjectHydrator extends AbstractHydrator continue; } - $parentClass = $this->ce[$this->_rsm->aliasMap[$parentAlias]]; + $parentClass = $this->_metadataCache[$this->_rsm->aliasMap[$parentAlias]]; $relationField = $this->_rsm->relationMap[$dqlAlias]; $relation = $parentClass->associationMappings[$relationField]; $reflField = $parentClass->reflFields[$relationField]; @@ -439,7 +431,7 @@ class ObjectHydrator extends AbstractHydrator $element = $this->getEntity($data, $dqlAlias); $reflField->setValue($parentObject, $element); $this->_uow->setOriginalEntityProperty($oid, $relationField, $element); - $targetClass = $this->ce[$relation['targetEntity']]; + $targetClass = $this->_metadataCache[$relation['targetEntity']]; if ($relation['isOwningSide']) { // TODO: Just check hints['fetched'] here? diff --git a/lib/Doctrine/ORM/Internal/Hydration/SimpleObjectHydrator.php b/lib/Doctrine/ORM/Internal/Hydration/SimpleObjectHydrator.php index 5f5b06a2a..124b099c7 100644 --- a/lib/Doctrine/ORM/Internal/Hydration/SimpleObjectHydrator.php +++ b/lib/Doctrine/ORM/Internal/Hydration/SimpleObjectHydrator.php @@ -32,9 +32,20 @@ class SimpleObjectHydrator extends AbstractHydrator private $class; /** - * @var array + * {@inheritdoc} */ - private $declaringClasses = array(); + protected function prepare() + { + if (count($this->_rsm->aliasMap) !== 1) { + throw new \RuntimeException("Cannot use SimpleObjectHydrator with a ResultSetMapping that contains more than one object result."); + } + + if ($this->_rsm->scalarMappings) { + throw new \RuntimeException("Cannot use SimpleObjectHydrator with a ResultSetMapping that contains scalar mappings."); + } + + $this->class = $this->getClassMetadata(reset($this->_rsm->aliasMap)); + } /** * {@inheritdoc} @@ -52,31 +63,6 @@ class SimpleObjectHydrator extends AbstractHydrator return $result; } - /** - * {@inheritdoc} - */ - protected function prepare() - { - if (count($this->_rsm->aliasMap) !== 1) { - throw new \RuntimeException("Cannot use SimpleObjectHydrator with a ResultSetMapping that contains more than one object result."); - } - - if ($this->_rsm->scalarMappings) { - throw new \RuntimeException("Cannot use SimpleObjectHydrator with a ResultSetMapping that contains scalar mappings."); - } - - $this->class = $this->_em->getClassMetadata(reset($this->_rsm->aliasMap)); - - // We only need to add declaring classes if we have inheritance. - if ($this->class->inheritanceType === ClassMetadata::INHERITANCE_TYPE_NONE) { - return; - } - - foreach ($this->_rsm->declaringClasses as $column => $class) { - $this->declaringClasses[$column] = $this->_em->getClassMetadata($class); - } - } - /** * {@inheritdoc} */