HHH-13654 Make AbstractFlushingEventListener#entitiesByKey also lazily initialized

This commit is contained in:
Sanne Grinovero 2019-10-30 10:27:12 +00:00
parent 6034ece731
commit f89bf35106
5 changed files with 77 additions and 71 deletions

View File

@ -95,7 +95,7 @@ public class StatefulPersistenceContext implements PersistenceContext {
private SharedSessionContractImplementor session; private SharedSessionContractImplementor session;
// Loaded entity instances, by EntityKey // Loaded entity instances, by EntityKey
private Map<EntityKey, Object> entitiesByKey; private HashMap<EntityKey, Object> entitiesByKey;
// Loaded entity instances, by EntityUniqueKey // Loaded entity instances, by EntityUniqueKey
private Map<EntityUniqueKey, Object> entitiesByUniqueKey; private Map<EntityUniqueKey, Object> entitiesByUniqueKey;
@ -155,8 +155,6 @@ public class StatefulPersistenceContext implements PersistenceContext {
*/ */
public StatefulPersistenceContext(SharedSessionContractImplementor session) { public StatefulPersistenceContext(SharedSessionContractImplementor session) {
this.session = session; this.session = session;
entitiesByKey = new HashMap<>( INIT_COLL_SIZE );
entityEntryContext = new EntityEntryContext( this ); entityEntryContext = new EntityEntryContext( this );
} }
@ -239,7 +237,7 @@ public class StatefulPersistenceContext implements PersistenceContext {
} }
arrayHolders = null; arrayHolders = null;
entitiesByKey.clear(); entitiesByKey = null;
entitiesByUniqueKey = null; entitiesByUniqueKey = null;
entityEntryContext.clear(); entityEntryContext.clear();
parentsByChild = null; parentsByChild = null;
@ -383,6 +381,9 @@ public class StatefulPersistenceContext implements PersistenceContext {
@Override @Override
public void addEntity(EntityKey key, Object entity) { public void addEntity(EntityKey key, Object entity) {
if ( entitiesByKey == null ) {
entitiesByKey = new HashMap<>( INIT_COLL_SIZE );
}
entitiesByKey.put( key, entity ); entitiesByKey.put( key, entity );
final BatchFetchQueue fetchQueue = this.batchFetchQueue; final BatchFetchQueue fetchQueue = this.batchFetchQueue;
if ( fetchQueue != null ) { if ( fetchQueue != null ) {
@ -392,17 +393,19 @@ public class StatefulPersistenceContext implements PersistenceContext {
@Override @Override
public Object getEntity(EntityKey key) { public Object getEntity(EntityKey key) {
return entitiesByKey.get( key ); return entitiesByKey == null ? null : entitiesByKey.get( key );
} }
@Override @Override
public boolean containsEntity(EntityKey key) { public boolean containsEntity(EntityKey key) {
return entitiesByKey.containsKey( key ); return entitiesByKey == null ? false : entitiesByKey.containsKey( key );
} }
@Override @Override
public Object removeEntity(EntityKey key) { public Object removeEntity(EntityKey key) {
final Object entity = entitiesByKey.remove( key ); final Object entity;
if ( entitiesByKey != null ) {
entity = entitiesByKey.remove( key );
if ( entitiesByUniqueKey != null ) { if ( entitiesByUniqueKey != null ) {
final Iterator itr = entitiesByUniqueKey.values().iterator(); final Iterator itr = entitiesByUniqueKey.values().iterator();
while ( itr.hasNext() ) { while ( itr.hasNext() ) {
@ -411,6 +414,10 @@ public class StatefulPersistenceContext implements PersistenceContext {
} }
} }
} }
}
else {
entity = null;
}
// Clear all parent cache // Clear all parent cache
parentsByChild = null; parentsByChild = null;
@ -757,6 +764,9 @@ public class StatefulPersistenceContext implements PersistenceContext {
@Override @Override
public void addEnhancedProxy(EntityKey key, PersistentAttributeInterceptable entity) { public void addEnhancedProxy(EntityKey key, PersistentAttributeInterceptable entity) {
if ( entitiesByKey == null ) {
entitiesByKey = new HashMap<>( INIT_COLL_SIZE );
}
entitiesByKey.put( key, entity ); entitiesByKey.put( key, entity );
} }
@ -1062,9 +1072,25 @@ public class StatefulPersistenceContext implements PersistenceContext {
return nullifiableEntityKeys; return nullifiableEntityKeys;
} }
/**
* @deprecated this will be removed: it provides too wide access, making it hard to optimise the internals
* for specific access needs. Consider using #iterateEntities instead.
* @return
*/
@Deprecated
@Override @Override
public Map getEntitiesByKey() { public Map getEntitiesByKey() {
return entitiesByKey; return entitiesByKey == null ? Collections.emptyMap() : entitiesByKey;
}
@Override
public Iterator managedEntitiesIterator() {
if ( entitiesByKey == null ) {
return Collections.emptyIterator();
}
else {
return entitiesByKey.values().iterator();
}
} }
@Override @Override
@ -1200,13 +1226,9 @@ public class StatefulPersistenceContext implements PersistenceContext {
@Override @Override
public String toString() { public String toString() {
if ( collectionsByKey == null ) { final String entityKeySet = entitiesByKey == null ? "[]" : entitiesByKey.keySet().toString();
return "PersistenceContext[entityKeys=" + entitiesByKey.keySet() + ",collectionKeys=[]]"; final String collectionsKeySet = collectionsByKey == null ? "[]" : collectionsByKey.keySet().toString();
} return "PersistenceContext[entityKeys=" + entityKeySet + ", collectionKeys=" + collectionsKeySet + "]";
else {
return "PersistenceContext[entityKeys=" + entitiesByKey.keySet()
+ ",collectionKeys=" + collectionsByKey.keySet() + "]";
}
} }
@Override @Override
@ -1503,7 +1525,7 @@ public class StatefulPersistenceContext implements PersistenceContext {
@Override @Override
public void replaceDelayedEntityIdentityInsertKeys(EntityKey oldKey, Serializable generatedId) { public void replaceDelayedEntityIdentityInsertKeys(EntityKey oldKey, Serializable generatedId) {
final Object entity = entitiesByKey.remove( oldKey ); final Object entity = entitiesByKey == null ? null : entitiesByKey.remove( oldKey );
final EntityEntry oldEntry = entityEntryContext.removeEntityEntry( entity ); final EntityEntry oldEntry = entityEntryContext.removeEntityEntry( entity );
this.parentsByChild = null; this.parentsByChild = null;
@ -1536,6 +1558,10 @@ public class StatefulPersistenceContext implements PersistenceContext {
oos.writeBoolean( defaultReadOnly ); oos.writeBoolean( defaultReadOnly );
oos.writeBoolean( hasNonReadOnlyEntities ); oos.writeBoolean( hasNonReadOnlyEntities );
if ( entitiesByKey == null ) {
oos.writeInt( 0 );
}
else {
oos.writeInt( entitiesByKey.size() ); oos.writeInt( entitiesByKey.size() );
if ( LOG.isTraceEnabled() ) { if ( LOG.isTraceEnabled() ) {
LOG.trace( "Starting serialization of [" + entitiesByKey.size() + "] entitiesByKey entries" ); LOG.trace( "Starting serialization of [" + entitiesByKey.size() + "] entitiesByKey entries" );
@ -1544,6 +1570,7 @@ public class StatefulPersistenceContext implements PersistenceContext {
entry.getKey().serialize( oos ); entry.getKey().serialize( oos );
oos.writeObject( entry.getValue() ); oos.writeObject( entry.getValue() );
} }
}
if ( entitiesByUniqueKey == null ) { if ( entitiesByUniqueKey == null ) {
oos.writeInt( 0 ); oos.writeInt( 0 );

View File

@ -9,6 +9,7 @@ package org.hibernate.engine.spi;
import java.io.Serializable; import java.io.Serializable;
import java.util.Collection; import java.util.Collection;
import java.util.HashSet; import java.util.HashSet;
import java.util.Iterator;
import java.util.Map; import java.util.Map;
import java.util.function.BiConsumer; import java.util.function.BiConsumer;
import java.util.function.Supplier; import java.util.function.Supplier;
@ -486,7 +487,10 @@ public interface PersistenceContext {
/** /**
* Get the mapping from key value to entity instance * Get the mapping from key value to entity instance
* @deprecated this will be removed: it provides too wide access, making it hard to optimise the internals
* for specific access needs. Consider using #iterateEntities instead.
*/ */
@Deprecated
Map getEntitiesByKey(); Map getEntitiesByKey();
/** /**
@ -793,6 +797,12 @@ public interface PersistenceContext {
*/ */
void removeCollectionByKey(CollectionKey collectionKey); void removeCollectionByKey(CollectionKey collectionKey);
/**
* A read-only iterator on all entities managed by this persistence context
* @return
*/
Iterator managedEntitiesIterator();
/** /**
* Provides centralized access to natural-id-related functionality. * Provides centralized access to natural-id-related functionality.
*/ */

View File

@ -37,7 +37,6 @@ import org.hibernate.event.spi.FlushEntityEventListener;
import org.hibernate.event.spi.FlushEvent; import org.hibernate.event.spi.FlushEvent;
import org.hibernate.internal.CoreMessageLogger; import org.hibernate.internal.CoreMessageLogger;
import org.hibernate.internal.util.EntityPrinter; import org.hibernate.internal.util.EntityPrinter;
import org.hibernate.internal.util.collections.LazyIterator;
import org.hibernate.persister.entity.EntityPersister; import org.hibernate.persister.entity.EntityPersister;
import org.jboss.logging.Logger; import org.jboss.logging.Logger;
@ -77,7 +76,7 @@ public abstract class AbstractFlushingEventListener implements JpaBootstrapSensi
EventSource session = event.getSession(); EventSource session = event.getSession();
final PersistenceContext persistenceContext = session.getPersistenceContextInternal(); final PersistenceContext persistenceContext = session.getPersistenceContextInternal();
session.getInterceptor().preFlush( new LazyIterator( persistenceContext.getEntitiesByKey() ) ); session.getInterceptor().preFlush( persistenceContext.managedEntitiesIterator() );
prepareEntityFlushes( session, persistenceContext ); prepareEntityFlushes( session, persistenceContext );
// we could move this inside if we wanted to // we could move this inside if we wanted to
@ -397,6 +396,7 @@ public abstract class AbstractFlushingEventListener implements JpaBootstrapSensi
} }
protected void postPostFlush(SessionImplementor session) { protected void postPostFlush(SessionImplementor session) {
session.getInterceptor().postFlush( new LazyIterator( session.getPersistenceContextInternal().getEntitiesByKey() ) ); session.getInterceptor().postFlush( session.getPersistenceContextInternal().managedEntitiesIterator() );
} }
} }

View File

@ -1,40 +0,0 @@
/*
* Hibernate, Relational Persistence for Idiomatic Java
*
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
* See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
*/
package org.hibernate.internal.util.collections;
import java.util.Iterator;
import java.util.Map;
public final class LazyIterator implements Iterator {
private final Map map;
private Iterator iterator;
private Iterator getIterator() {
if (iterator==null) {
iterator = map.values().iterator();
}
return iterator;
}
public LazyIterator(Map map) {
this.map = map;
}
public boolean hasNext() {
return getIterator().hasNext();
}
public Object next() {
return getIterator().next();
}
public void remove() {
throw new UnsupportedOperationException();
}
}

View File

@ -6,6 +6,7 @@
*/ */
package org.hibernate.test.stateless; package org.hibernate.test.stateless;
import java.util.Collections;
import java.util.function.Consumer; import java.util.function.Consumer;
import javax.persistence.Entity; import javax.persistence.Entity;
import javax.persistence.GeneratedValue; import javax.persistence.GeneratedValue;
@ -115,10 +116,18 @@ public class StatelessSessionPersistentContextTest extends BaseCoreFunctionalTes
"StatelessSession: PersistenceContext has not been cleared", "StatelessSession: PersistenceContext has not been cleared",
persistenceContextInternal.getEntitiesByKey().isEmpty() persistenceContextInternal.getEntitiesByKey().isEmpty()
); );
assertTrue(
"StatelessSession: PersistenceContext has not been cleared",
persistenceContextInternal.managedEntitiesIterator() == Collections.emptyIterator()
);
assertTrue( assertTrue(
"StatelessSession: PersistenceContext has not been cleared", "StatelessSession: PersistenceContext has not been cleared",
persistenceContextInternal.getCollectionsByKey().isEmpty() persistenceContextInternal.getCollectionsByKey().isEmpty()
); );
assertTrue(
"StatelessSession: PersistenceContext has not been cleared",
persistenceContextInternal.getCollectionsByKey() == Collections.emptyMap()
);
} }
@Entity(name = "TestEntity") @Entity(name = "TestEntity")