HHH-15569 Replace list and map with array to improve state extraction for entity initializing
This commit is contained in:
parent
1d5f6b5c13
commit
99f9ccdd11
|
@ -6,6 +6,7 @@
|
||||||
*/
|
*/
|
||||||
package org.hibernate.metamodel.mapping;
|
package org.hibernate.metamodel.mapping;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
@ -23,6 +24,7 @@ import org.hibernate.loader.ast.spi.NaturalIdLoader;
|
||||||
import org.hibernate.mapping.IndexedConsumer;
|
import org.hibernate.mapping.IndexedConsumer;
|
||||||
import org.hibernate.metamodel.UnsupportedMappingException;
|
import org.hibernate.metamodel.UnsupportedMappingException;
|
||||||
import org.hibernate.metamodel.spi.EntityRepresentationStrategy;
|
import org.hibernate.metamodel.spi.EntityRepresentationStrategy;
|
||||||
|
import org.hibernate.metamodel.spi.MappingMetamodelImplementor;
|
||||||
import org.hibernate.persister.entity.EntityPersister;
|
import org.hibernate.persister.entity.EntityPersister;
|
||||||
import org.hibernate.spi.NavigablePath;
|
import org.hibernate.spi.NavigablePath;
|
||||||
import org.hibernate.query.sqm.mutation.spi.SqmMultiTableInsertStrategy;
|
import org.hibernate.query.sqm.mutation.spi.SqmMultiTableInsertStrategy;
|
||||||
|
@ -167,6 +169,10 @@ public interface EntityMappingType extends ManagedMappingType, EntityValuedModel
|
||||||
// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
// Inheritance
|
// Inheritance
|
||||||
|
|
||||||
|
default int getSubclassId() {
|
||||||
|
return getEntityPersister().getEntityMetamodel().getSubclassId();
|
||||||
|
}
|
||||||
|
|
||||||
default boolean hasSubclasses() {
|
default boolean hasSubclasses() {
|
||||||
return getEntityPersister().getEntityMetamodel().hasSubclasses();
|
return getEntityPersister().getEntityMetamodel().hasSubclasses();
|
||||||
}
|
}
|
||||||
|
@ -198,6 +204,16 @@ public interface EntityMappingType extends ManagedMappingType, EntityValuedModel
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
default Collection<EntityMappingType> getSubMappingTypes() {
|
||||||
|
final MappingMetamodelImplementor mappingMetamodel = getEntityPersister().getFactory().getMappingMetamodel();
|
||||||
|
final Set<String> subclassEntityNames = getSubclassEntityNames();
|
||||||
|
final List<EntityMappingType> mappingTypes = new ArrayList<>( subclassEntityNames.size() );
|
||||||
|
for ( String subclassEntityName : subclassEntityNames ) {
|
||||||
|
mappingTypes.add( mappingMetamodel.getEntityDescriptor( subclassEntityName ) );
|
||||||
|
}
|
||||||
|
return mappingTypes;
|
||||||
|
}
|
||||||
|
|
||||||
default boolean isTypeOrSuperType(EntityMappingType targetType) {
|
default boolean isTypeOrSuperType(EntityMappingType targetType) {
|
||||||
return targetType == this;
|
return targetType == this;
|
||||||
}
|
}
|
||||||
|
@ -314,6 +330,7 @@ public interface EntityMappingType extends ManagedMappingType, EntityValuedModel
|
||||||
|
|
||||||
// Customer <- DomesticCustomer <- OtherCustomer
|
// Customer <- DomesticCustomer <- OtherCustomer
|
||||||
|
|
||||||
|
@Deprecated(forRemoval = true)
|
||||||
default Object[] extractConcreteTypeStateValues(
|
default Object[] extractConcreteTypeStateValues(
|
||||||
Map<AttributeMapping, DomainResultAssembler> assemblerMapping,
|
Map<AttributeMapping, DomainResultAssembler> assemblerMapping,
|
||||||
RowProcessingState rowProcessingState) {
|
RowProcessingState rowProcessingState) {
|
||||||
|
|
|
@ -1221,7 +1221,7 @@ public abstract class AbstractEntityPersister
|
||||||
// of this class, not its subclasses, but since we
|
// of this class, not its subclasses, but since we
|
||||||
// are reusing code used for sequential selects, we
|
// are reusing code used for sequential selects, we
|
||||||
// use the subclass closure
|
// use the subclass closure
|
||||||
partsToSelect.add( getAttributeMappings().get( getSubclassPropertyIndex( lazyAttributeDescriptor.getName() ) ) );
|
partsToSelect.add( getAttributeMapping( getSubclassPropertyIndex( lazyAttributeDescriptor.getName() ) ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( partsToSelect.isEmpty() ) {
|
if ( partsToSelect.isEmpty() ) {
|
||||||
|
@ -2915,8 +2915,8 @@ public abstract class AbstractEntityPersister
|
||||||
final Update update = createUpdate().setTableName( getTableName( j ) );
|
final Update update = createUpdate().setTableName( getTableName( j ) );
|
||||||
|
|
||||||
boolean hasColumns = false;
|
boolean hasColumns = false;
|
||||||
for ( int index = 0; index < attributeMappings.size(); index++ ) {
|
for ( int index = 0; index < attributeMappings.length; index++ ) {
|
||||||
final AttributeMapping attributeMapping = attributeMappings.get( index );
|
final AttributeMapping attributeMapping = attributeMappings[index];
|
||||||
if ( isPropertyOfTable( index, j ) ) {
|
if ( isPropertyOfTable( index, j ) ) {
|
||||||
// `attributeMapping` is an attribute of the table we are updating
|
// `attributeMapping` is an attribute of the table we are updating
|
||||||
|
|
||||||
|
@ -3039,8 +3039,8 @@ public abstract class AbstractEntityPersister
|
||||||
|
|
||||||
final Insert insert = createInsert().setTableName( getTableName( j ) );
|
final Insert insert = createInsert().setTableName( getTableName( j ) );
|
||||||
|
|
||||||
for ( int index = 0; index < attributeMappings.size(); index++ ) {
|
for ( int index = 0; index < attributeMappings.length; index++ ) {
|
||||||
final AttributeMapping attributeMapping = attributeMappings.get( index );
|
final AttributeMapping attributeMapping = attributeMappings[index];
|
||||||
if ( isPropertyOfTable( index, j ) ) {
|
if ( isPropertyOfTable( index, j ) ) {
|
||||||
// `attributeMapping` is an attribute of the table we are updating
|
// `attributeMapping` is an attribute of the table we are updating
|
||||||
|
|
||||||
|
@ -5117,10 +5117,10 @@ public abstract class AbstractEntityPersister
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if ( hasSubclasses() ) {
|
if ( hasSubclasses() ) {
|
||||||
for ( int i = 0; i < attributeMappings.size(); i++ ) {
|
for ( int i = 0; i < attributeMappings.length; i++ ) {
|
||||||
final Object value = values[i];
|
final Object value = values[i];
|
||||||
if ( value != UNFETCHED_PROPERTY ) {
|
if ( value != UNFETCHED_PROPERTY ) {
|
||||||
final Setter setter = attributeMappings.get( i ).getPropertyAccess().getSetter();
|
final Setter setter = attributeMappings[i].getPropertyAccess().getSetter();
|
||||||
setter.set( object, value );
|
setter.set( object, value );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5153,8 +5153,8 @@ public abstract class AbstractEntityPersister
|
||||||
final BytecodeEnhancementMetadata enhancementMetadata = entityMetamodel.getBytecodeEnhancementMetadata();
|
final BytecodeEnhancementMetadata enhancementMetadata = entityMetamodel.getBytecodeEnhancementMetadata();
|
||||||
final LazyAttributesMetadata lazyAttributesMetadata = enhancementMetadata.getLazyAttributesMetadata();
|
final LazyAttributesMetadata lazyAttributesMetadata = enhancementMetadata.getLazyAttributesMetadata();
|
||||||
final Object[] values = new Object[ getNumberOfAttributeMappings() ];
|
final Object[] values = new Object[ getNumberOfAttributeMappings() ];
|
||||||
for ( int i = 0; i < attributeMappings.size(); i++ ) {
|
for ( int i = 0; i < attributeMappings.length; i++ ) {
|
||||||
final AttributeMapping attributeMapping = attributeMappings.get( i );
|
final AttributeMapping attributeMapping = attributeMappings[i];
|
||||||
final AttributeMetadataAccess attributeMetadataAccess = attributeMapping.getAttributeMetadataAccess();
|
final AttributeMetadataAccess attributeMetadataAccess = attributeMapping.getAttributeMetadataAccess();
|
||||||
if ( ! lazyAttributesMetadata.isLazyAttribute( attributeMapping.getAttributeName() )
|
if ( ! lazyAttributesMetadata.isLazyAttribute( attributeMapping.getAttributeName() )
|
||||||
|| enhancementMetadata.isAttributeLoaded( object, attributeMapping.getAttributeName() ) ) {
|
|| enhancementMetadata.isAttributeLoaded( object, attributeMapping.getAttributeName() ) ) {
|
||||||
|
@ -5175,7 +5175,7 @@ public abstract class AbstractEntityPersister
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Object getPropertyValue(Object object, int i) {
|
public Object getPropertyValue(Object object, int i) {
|
||||||
return attributeMappings.get( i ).getAttributeMetadataAccess()
|
return attributeMappings[i].getAttributeMetadataAccess()
|
||||||
.resolveAttributeMetadata( this )
|
.resolveAttributeMetadata( this )
|
||||||
.getPropertyAccess()
|
.getPropertyAccess()
|
||||||
.getGetter()
|
.getGetter()
|
||||||
|
@ -5370,9 +5370,9 @@ public abstract class AbstractEntityPersister
|
||||||
return accessOptimizer.getPropertyValues( entity );
|
return accessOptimizer.getPropertyValues( entity );
|
||||||
}
|
}
|
||||||
|
|
||||||
final Object[] result = new Object[this.attributeMappings.size()];
|
final Object[] result = new Object[this.attributeMappings.length];
|
||||||
for ( int i = 0; i < this.attributeMappings.size(); i++ ) {
|
for ( int i = 0; i < this.attributeMappings.length; i++ ) {
|
||||||
result[i] = this.attributeMappings.get( i ).getPropertyAccess().getGetter().getForInsert(
|
result[i] = this.attributeMappings[i].getPropertyAccess().getGetter().getForInsert(
|
||||||
entity,
|
entity,
|
||||||
mergeMap,
|
mergeMap,
|
||||||
session
|
session
|
||||||
|
@ -5668,7 +5668,7 @@ public abstract class AbstractEntityPersister
|
||||||
private EntityRowIdMapping rowIdMapping;
|
private EntityRowIdMapping rowIdMapping;
|
||||||
private EntityDiscriminatorMapping discriminatorMapping;
|
private EntityDiscriminatorMapping discriminatorMapping;
|
||||||
|
|
||||||
private List<AttributeMapping> attributeMappings;
|
private AttributeMapping[] attributeMappings;
|
||||||
protected Map<String, AttributeMapping> declaredAttributeMappings = new LinkedHashMap<>();
|
protected Map<String, AttributeMapping> declaredAttributeMappings = new LinkedHashMap<>();
|
||||||
protected List<Fetchable> staticFetchableList;
|
protected List<Fetchable> staticFetchableList;
|
||||||
|
|
||||||
|
@ -5676,13 +5676,15 @@ public abstract class AbstractEntityPersister
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visitAttributeMappings(Consumer<? super AttributeMapping> action) {
|
public void visitAttributeMappings(Consumer<? super AttributeMapping> action) {
|
||||||
attributeMappings.forEach( action );
|
for ( AttributeMapping attributeMapping : attributeMappings ) {
|
||||||
|
action.accept( attributeMapping );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void forEachAttributeMapping(IndexedConsumer<AttributeMapping> consumer) {
|
public void forEachAttributeMapping(IndexedConsumer<AttributeMapping> consumer) {
|
||||||
for ( int i = 0; i < attributeMappings.size(); i++ ) {
|
for ( int i = 0; i < attributeMappings.length; i++ ) {
|
||||||
consumer.accept( i, attributeMappings.get( i ) );
|
consumer.accept( i, attributeMappings[i] );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5770,7 +5772,7 @@ public abstract class AbstractEntityPersister
|
||||||
if ( hasUpdateGeneratedProperties() ) {
|
if ( hasUpdateGeneratedProperties() ) {
|
||||||
updateGeneratedValuesProcessor = createGeneratedValuesProcessor( GenerationTiming.UPDATE );
|
updateGeneratedValuesProcessor = createGeneratedValuesProcessor( GenerationTiming.UPDATE );
|
||||||
}
|
}
|
||||||
staticFetchableList = new ArrayList<>( attributeMappings.size() );
|
staticFetchableList = new ArrayList<>( attributeMappings.length );
|
||||||
visitSubTypeAttributeMappings( attributeMapping -> staticFetchableList.add( attributeMapping ) );
|
visitSubTypeAttributeMappings( attributeMapping -> staticFetchableList.add( attributeMapping ) );
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -5910,13 +5912,11 @@ public abstract class AbstractEntityPersister
|
||||||
// in the collected names. iterate here because it is already alphabetical
|
// in the collected names. iterate here because it is already alphabetical
|
||||||
|
|
||||||
final List<SingularAttributeMapping> collectedAttrMappings = new ArrayList<>();
|
final List<SingularAttributeMapping> collectedAttrMappings = new ArrayList<>();
|
||||||
this.attributeMappings.forEach(
|
for ( AttributeMapping attributeMapping : attributeMappings ) {
|
||||||
(attributeMapping) -> {
|
if ( attributeNames.contains( attributeMapping.getAttributeName() ) ) {
|
||||||
if ( attributeNames.contains( attributeMapping.getAttributeName() ) ) {
|
collectedAttrMappings.add( (SingularAttributeMapping) attributeMapping );
|
||||||
collectedAttrMappings.add( (SingularAttributeMapping) attributeMapping );
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
if ( collectedAttrMappings.size() <= 1 ) {
|
if ( collectedAttrMappings.size() <= 1 ) {
|
||||||
throw new MappingException( "Expected multiple natural-id attributes, but found only one: " + getEntityName() );
|
throw new MappingException( "Expected multiple natural-id attributes, but found only one: " + getEntityName() );
|
||||||
|
@ -6106,12 +6106,12 @@ public abstract class AbstractEntityPersister
|
||||||
// force calculation of `attributeMappings`
|
// force calculation of `attributeMappings`
|
||||||
getAttributeMappings();
|
getAttributeMappings();
|
||||||
}
|
}
|
||||||
return attributeMappings.size();
|
return attributeMappings.length;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public AttributeMapping getAttributeMapping(int position) {
|
public AttributeMapping getAttributeMapping(int position) {
|
||||||
return attributeMappings.get( position );
|
return attributeMappings[position];
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -6134,6 +6134,14 @@ public abstract class AbstractEntityPersister
|
||||||
return superMappingType;
|
return superMappingType;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Collection<EntityMappingType> getSubMappingTypes() {
|
||||||
|
if ( subclassMappingTypes == null ) {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
return subclassMappingTypes.values();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean isTypeOrSuperType(EntityMappingType targetType) {
|
public boolean isTypeOrSuperType(EntityMappingType targetType) {
|
||||||
if ( targetType == null ) {
|
if ( targetType == null ) {
|
||||||
|
@ -6534,18 +6542,18 @@ public abstract class AbstractEntityPersister
|
||||||
@Override
|
@Override
|
||||||
public List<AttributeMapping> getAttributeMappings() {
|
public List<AttributeMapping> getAttributeMappings() {
|
||||||
if ( attributeMappings == null ) {
|
if ( attributeMappings == null ) {
|
||||||
attributeMappings = new ArrayList<>();
|
ArrayList<AttributeMapping> attributeMappings = new ArrayList<>();
|
||||||
|
|
||||||
if ( superMappingType != null ) {
|
if ( superMappingType != null ) {
|
||||||
superMappingType.visitAttributeMappings( attributeMappings::add );
|
superMappingType.visitAttributeMappings( attributeMappings::add );
|
||||||
}
|
}
|
||||||
|
|
||||||
attributeMappings.addAll( declaredAttributeMappings.values() );
|
attributeMappings.addAll( declaredAttributeMappings.values() );
|
||||||
|
this.attributeMappings = attributeMappings.toArray(new AttributeMapping[0]);
|
||||||
// subclasses? it depends on the usage
|
// subclasses? it depends on the usage
|
||||||
}
|
}
|
||||||
|
|
||||||
return attributeMappings;
|
return Arrays.asList( attributeMappings );
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -6769,7 +6777,9 @@ public abstract class AbstractEntityPersister
|
||||||
visitSubTypeAttributeMappings( fetchableConsumer );
|
visitSubTypeAttributeMappings( fetchableConsumer );
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
attributeMappings.forEach( fetchableConsumer );
|
for ( AttributeMapping attributeMapping : attributeMappings ) {
|
||||||
|
fetchableConsumer.accept( attributeMapping );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -6785,9 +6795,9 @@ public abstract class AbstractEntityPersister
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final int size = attributeMappings.size();
|
final int size = attributeMappings.length;
|
||||||
for ( int i = 0; i < size; i++ ) {
|
for ( int i = 0; i < size; i++ ) {
|
||||||
fetchableConsumer.accept( i, attributeMappings.get( i ) );
|
fetchableConsumer.accept( i, attributeMappings[i] );
|
||||||
}
|
}
|
||||||
if ( treatTargetType.isTypeOrSuperType( this ) ) {
|
if ( treatTargetType.isTypeOrSuperType( this ) ) {
|
||||||
if ( subclassMappingTypes != null ) {
|
if ( subclassMappingTypes != null ) {
|
||||||
|
@ -6810,7 +6820,9 @@ public abstract class AbstractEntityPersister
|
||||||
public void visitAttributeMappings(
|
public void visitAttributeMappings(
|
||||||
Consumer<? super AttributeMapping> action,
|
Consumer<? super AttributeMapping> action,
|
||||||
EntityMappingType targetType) {
|
EntityMappingType targetType) {
|
||||||
attributeMappings.forEach( action );
|
for ( AttributeMapping attributeMapping : attributeMappings ) {
|
||||||
|
action.accept( attributeMapping );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -6823,9 +6835,8 @@ public abstract class AbstractEntityPersister
|
||||||
@Override
|
@Override
|
||||||
public int forEachSelectable(int offset, SelectableConsumer selectableConsumer) {
|
public int forEachSelectable(int offset, SelectableConsumer selectableConsumer) {
|
||||||
int span = 0;
|
int span = 0;
|
||||||
final List<AttributeMapping> mappings = getAttributeMappings();
|
for ( AttributeMapping attributeMapping : attributeMappings ) {
|
||||||
for ( int i = 0; i < mappings.size(); i++ ) {
|
span += attributeMapping.forEachSelectable( span + offset, selectableConsumer );
|
||||||
span += mappings.get( i ).forEachSelectable( span + offset, selectableConsumer );
|
|
||||||
}
|
}
|
||||||
return span;
|
return span;
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,8 +6,7 @@
|
||||||
*/
|
*/
|
||||||
package org.hibernate.sql.results.graph.entity;
|
package org.hibernate.sql.results.graph.entity;
|
||||||
|
|
||||||
import java.util.IdentityHashMap;
|
import java.util.Collection;
|
||||||
import java.util.Map;
|
|
||||||
import java.util.function.Consumer;
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
import org.hibernate.HibernateException;
|
import org.hibernate.HibernateException;
|
||||||
|
@ -33,9 +32,10 @@ import org.hibernate.event.spi.PreLoadEvent;
|
||||||
import org.hibernate.event.spi.PreLoadEventListener;
|
import org.hibernate.event.spi.PreLoadEventListener;
|
||||||
import org.hibernate.internal.util.NullnessHelper;
|
import org.hibernate.internal.util.NullnessHelper;
|
||||||
import org.hibernate.internal.util.StringHelper;
|
import org.hibernate.internal.util.StringHelper;
|
||||||
import org.hibernate.internal.util.collections.CollectionHelper;
|
|
||||||
import org.hibernate.loader.entity.CacheEntityLoaderHelper;
|
import org.hibernate.loader.entity.CacheEntityLoaderHelper;
|
||||||
import org.hibernate.metamodel.mapping.AttributeMapping;
|
import org.hibernate.metamodel.mapping.AttributeMapping;
|
||||||
|
import org.hibernate.metamodel.mapping.AttributeMetadata;
|
||||||
|
import org.hibernate.metamodel.mapping.EntityMappingType;
|
||||||
import org.hibernate.metamodel.mapping.EntityValuedModelPart;
|
import org.hibernate.metamodel.mapping.EntityValuedModelPart;
|
||||||
import org.hibernate.metamodel.mapping.EntityVersionMapping;
|
import org.hibernate.metamodel.mapping.EntityVersionMapping;
|
||||||
import org.hibernate.metamodel.mapping.ManagedMappingType;
|
import org.hibernate.metamodel.mapping.ManagedMappingType;
|
||||||
|
@ -64,6 +64,7 @@ import org.hibernate.type.Type;
|
||||||
|
|
||||||
import static org.hibernate.engine.internal.ManagedTypeHelper.asPersistentAttributeInterceptable;
|
import static org.hibernate.engine.internal.ManagedTypeHelper.asPersistentAttributeInterceptable;
|
||||||
import static org.hibernate.engine.internal.ManagedTypeHelper.isPersistentAttributeInterceptable;
|
import static org.hibernate.engine.internal.ManagedTypeHelper.isPersistentAttributeInterceptable;
|
||||||
|
import static org.hibernate.bytecode.enhance.spi.LazyPropertyInitializer.UNFETCHED_PROPERTY;
|
||||||
import static org.hibernate.internal.log.LoggingHelper.toLoggableString;
|
import static org.hibernate.internal.log.LoggingHelper.toLoggableString;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -90,7 +91,7 @@ public abstract class AbstractEntityInitializer extends AbstractFetchParentAcces
|
||||||
private final DomainResultAssembler versionAssembler;
|
private final DomainResultAssembler versionAssembler;
|
||||||
private final DomainResultAssembler<Object> rowIdAssembler;
|
private final DomainResultAssembler<Object> rowIdAssembler;
|
||||||
|
|
||||||
private final Map<AttributeMapping, DomainResultAssembler> assemblerMap;
|
private final DomainResultAssembler[][] assemblers;
|
||||||
|
|
||||||
// per-row state
|
// per-row state
|
||||||
private EntityPersister concreteDescriptor;
|
private EntityPersister concreteDescriptor;
|
||||||
|
@ -167,17 +168,21 @@ public abstract class AbstractEntityInitializer extends AbstractFetchParentAcces
|
||||||
this.rowIdAssembler = null;
|
this.rowIdAssembler = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
final Collection<EntityMappingType> subMappingTypes = rootEntityDescriptor.getSubMappingTypes();
|
||||||
|
assemblers = new DomainResultAssembler[subMappingTypes.size() + 1][];
|
||||||
|
assemblers[rootEntityDescriptor.getSubclassId()] = new DomainResultAssembler[rootEntityDescriptor.getNumberOfFetchables()];
|
||||||
|
|
||||||
|
for ( EntityMappingType subMappingType : subMappingTypes ) {
|
||||||
|
assemblers[subMappingType.getSubclassId()] = new DomainResultAssembler[subMappingType.getNumberOfFetchables()];
|
||||||
|
}
|
||||||
|
|
||||||
final int size = entityDescriptor.getNumberOfFetchables();
|
final int size = entityDescriptor.getNumberOfFetchables();
|
||||||
assemblerMap = new IdentityHashMap<>( CollectionHelper.determineProperSizing( size ) );
|
|
||||||
for ( int i = 0; i < size; i++ ) {
|
for ( int i = 0; i < size; i++ ) {
|
||||||
final AttributeMapping attributeMapping = (AttributeMapping) entityDescriptor.getFetchable( i );
|
final AttributeMapping attributeMapping = (AttributeMapping) entityDescriptor.getFetchable( i );
|
||||||
|
|
||||||
// todo (6.0) : somehow we need to track whether all state is loaded/resolved
|
// todo (6.0) : somehow we need to track whether all state is loaded/resolved
|
||||||
// note that lazy proxies or uninitialized collections count against
|
// note that lazy proxies or uninitialized collections count against
|
||||||
// that in the affirmative
|
// that in the affirmative
|
||||||
|
|
||||||
final Fetch fetch = resultDescriptor.findFetch( attributeMapping );
|
final Fetch fetch = resultDescriptor.findFetch( attributeMapping );
|
||||||
|
|
||||||
final DomainResultAssembler<?> stateAssembler;
|
final DomainResultAssembler<?> stateAssembler;
|
||||||
if ( fetch == null ) {
|
if ( fetch == null ) {
|
||||||
stateAssembler = new NullValueAssembler<>(
|
stateAssembler = new NullValueAssembler<>(
|
||||||
|
@ -188,7 +193,12 @@ public abstract class AbstractEntityInitializer extends AbstractFetchParentAcces
|
||||||
stateAssembler = fetch.createAssembler( this, creationState );
|
stateAssembler = fetch.createAssembler( this, creationState );
|
||||||
}
|
}
|
||||||
|
|
||||||
assemblerMap.put( attributeMapping, stateAssembler );
|
final int stateArrayPosition = attributeMapping.getStateArrayPosition();
|
||||||
|
final EntityMappingType declaringType = (EntityMappingType) attributeMapping.getDeclaringType();
|
||||||
|
assemblers[declaringType.getSubclassId()][stateArrayPosition] = stateAssembler;
|
||||||
|
for ( EntityMappingType subMappingType : declaringType.getSubMappingTypes() ) {
|
||||||
|
assemblers[subMappingType.getSubclassId()][stateArrayPosition] = stateAssembler;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -197,25 +207,24 @@ public abstract class AbstractEntityInitializer extends AbstractFetchParentAcces
|
||||||
Object[] source,
|
Object[] source,
|
||||||
Object[] target,
|
Object[] target,
|
||||||
EntityPersister concreteDescriptor) {
|
EntityPersister concreteDescriptor) {
|
||||||
containerDescriptor.visitAttributeMappings(
|
final int numberOfAttributeMappings = containerDescriptor.getNumberOfAttributeMappings();
|
||||||
attributeMapping -> {
|
for ( int i = 0; i < numberOfAttributeMappings; i++ ) {
|
||||||
if ( attributeMapping.getAttributeMetadataAccess().resolveAttributeMetadata( concreteDescriptor ).isUpdatable() ) {
|
final AttributeMapping attributeMapping = containerDescriptor.getAttributeMapping( i );
|
||||||
final int position = attributeMapping.getStateArrayPosition();
|
final AttributeMetadata attributeMetadata = attributeMapping.getAttributeMetadataAccess()
|
||||||
Object result;
|
.resolveAttributeMetadata( concreteDescriptor );
|
||||||
if ( source[position] == LazyPropertyInitializer.UNFETCHED_PROPERTY
|
if ( attributeMetadata.isUpdatable() ) {
|
||||||
|| source[position] == PropertyAccessStrategyBackRefImpl.UNKNOWN ) {
|
final int position = attributeMapping.getStateArrayPosition();
|
||||||
result = source[position];
|
Object result;
|
||||||
}
|
if ( source[position] == LazyPropertyInitializer.UNFETCHED_PROPERTY
|
||||||
else {
|
|| source[position] == PropertyAccessStrategyBackRefImpl.UNKNOWN ) {
|
||||||
result = attributeMapping.getAttributeMetadataAccess()
|
result = source[position];
|
||||||
.resolveAttributeMetadata(null)
|
|
||||||
.getMutabilityPlan()
|
|
||||||
.deepCopy( source[position] );
|
|
||||||
}
|
|
||||||
target[position] = result;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
);
|
else {
|
||||||
|
result = attributeMetadata.getMutabilityPlan().deepCopy( source[position] );
|
||||||
|
}
|
||||||
|
target[position] = result;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -702,10 +711,7 @@ public abstract class AbstractEntityInitializer extends AbstractFetchParentAcces
|
||||||
|
|
||||||
entityDescriptor.setIdentifier( toInitialize, entityIdentifier, session );
|
entityDescriptor.setIdentifier( toInitialize, entityIdentifier, session );
|
||||||
|
|
||||||
resolvedEntityState = concreteDescriptor.extractConcreteTypeStateValues(
|
resolvedEntityState = extractConcreteTypeStateValues( rowProcessingState );
|
||||||
assemblerMap,
|
|
||||||
rowProcessingState
|
|
||||||
);
|
|
||||||
|
|
||||||
if ( isPersistentAttributeInterceptable( toInitialize ) ) {
|
if ( isPersistentAttributeInterceptable( toInitialize ) ) {
|
||||||
PersistentAttributeInterceptor persistentAttributeInterceptor = asPersistentAttributeInterceptable( toInitialize ).$$_hibernate_getInterceptor();
|
PersistentAttributeInterceptor persistentAttributeInterceptor = asPersistentAttributeInterceptable( toInitialize ).$$_hibernate_getInterceptor();
|
||||||
|
@ -900,6 +906,24 @@ public abstract class AbstractEntityInitializer extends AbstractFetchParentAcces
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Object[] extractConcreteTypeStateValues(RowProcessingState rowProcessingState) {
|
||||||
|
final Object[] values = new Object[concreteDescriptor.getNumberOfAttributeMappings()];
|
||||||
|
final DomainResultAssembler[] concreteAssemblers = assemblers[concreteDescriptor.getSubclassId()];
|
||||||
|
for ( int i = 0; i < values.length; i++ ) {
|
||||||
|
final DomainResultAssembler assembler = concreteAssemblers[i];
|
||||||
|
final Object value;
|
||||||
|
if ( assembler == null ) {
|
||||||
|
value = UNFETCHED_PROPERTY;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
value = assembler.assemble( rowProcessingState );
|
||||||
|
}
|
||||||
|
|
||||||
|
values[i] = value;
|
||||||
|
}
|
||||||
|
return values;
|
||||||
|
}
|
||||||
|
|
||||||
private boolean skipInitialization(
|
private boolean skipInitialization(
|
||||||
Object toInitialize,
|
Object toInitialize,
|
||||||
RowProcessingState rowProcessingState,
|
RowProcessingState rowProcessingState,
|
||||||
|
|
|
@ -74,6 +74,7 @@ public class EntityMetamodel implements Serializable {
|
||||||
private final String rootName;
|
private final String rootName;
|
||||||
private EntityType entityType;
|
private EntityType entityType;
|
||||||
|
|
||||||
|
private final int subclassId;
|
||||||
private final IdentifierProperty identifierAttribute;
|
private final IdentifierProperty identifierAttribute;
|
||||||
private final boolean versioned;
|
private final boolean versioned;
|
||||||
|
|
||||||
|
@ -151,6 +152,7 @@ public class EntityMetamodel implements Serializable {
|
||||||
// Improves performance of EntityKey#equals by avoiding content check in String#equals
|
// Improves performance of EntityKey#equals by avoiding content check in String#equals
|
||||||
name = persistentClass.getEntityName().intern();
|
name = persistentClass.getEntityName().intern();
|
||||||
rootName = persistentClass.getRootClass().getEntityName().intern();
|
rootName = persistentClass.getRootClass().getEntityName().intern();
|
||||||
|
subclassId = persistentClass.getSubclassId();
|
||||||
|
|
||||||
identifierAttribute = PropertyFactory.buildIdentifierAttribute(
|
identifierAttribute = PropertyFactory.buildIdentifierAttribute(
|
||||||
persistentClass,
|
persistentClass,
|
||||||
|
@ -892,6 +894,10 @@ public class EntityMetamodel implements Serializable {
|
||||||
return rootName;
|
return rootName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int getSubclassId() {
|
||||||
|
return subclassId;
|
||||||
|
}
|
||||||
|
|
||||||
public EntityType getEntityType() {
|
public EntityType getEntityType() {
|
||||||
if ( entityType == null ) {
|
if ( entityType == null ) {
|
||||||
entityType = new ManyToOneType( name, getSessionFactory().getTypeConfiguration() );
|
entityType = new ManyToOneType( name, getSessionFactory().getTypeConfiguration() );
|
||||||
|
|
Loading…
Reference in New Issue