HHH-10366 Improve performance and reduce CPU load when fetching reference cached objects
Removed a call to postLoad and afterInitialize for reference cached entities; refactor methods, made some private, some final try to optimize cached result refactor out getPersister so it can be inlined
This commit is contained in:
parent
4b4164f8e0
commit
c802042f1a
|
@ -182,37 +182,50 @@ public class StandardQueryCache implements QueryCache {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
final List result = new ArrayList( cacheable.size() - 1 );
|
return assembleCachedResult(key, cacheable, isNaturalKeyLookup, singleResult, returnTypes, session);
|
||||||
for ( int i = 1; i < cacheable.size(); i++ ) {
|
}
|
||||||
try {
|
|
||||||
if ( singleResult ) {
|
private List assembleCachedResult(
|
||||||
|
final QueryKey key,
|
||||||
|
final List cacheable,
|
||||||
|
final boolean isNaturalKeyLookup,
|
||||||
|
boolean singleResult,
|
||||||
|
final Type[] returnTypes,
|
||||||
|
final SessionImplementor session) throws HibernateException {
|
||||||
|
|
||||||
|
try {
|
||||||
|
final List result = new ArrayList( cacheable.size() - 1 );
|
||||||
|
if ( singleResult ) {
|
||||||
|
for ( int i = 1; i < cacheable.size(); i++ ) {
|
||||||
result.add( returnTypes[0].assemble( (Serializable) cacheable.get( i ), session, null ) );
|
result.add( returnTypes[0].assemble( (Serializable) cacheable.get( i ), session, null ) );
|
||||||
}
|
}
|
||||||
else {
|
}
|
||||||
|
else {
|
||||||
|
for ( int i = 1; i < cacheable.size(); i++ ) {
|
||||||
result.add(
|
result.add(
|
||||||
TypeHelper.assemble( (Serializable[]) cacheable.get( i ), returnTypes, session, null )
|
TypeHelper.assemble( (Serializable[]) cacheable.get( i ), returnTypes, session, null )
|
||||||
);
|
);
|
||||||
|
logCachedResultRowDetails( returnTypes, result.get( i - 1 ) );
|
||||||
}
|
}
|
||||||
logCachedResultRowDetails( returnTypes, result.get( i - 1 ) );
|
|
||||||
}
|
|
||||||
catch ( RuntimeException ex ) {
|
|
||||||
if ( isNaturalKeyLookup ) {
|
|
||||||
// potentially perform special handling for natural-id look ups.
|
|
||||||
if ( UnresolvableObjectException.class.isInstance( ex )
|
|
||||||
|| EntityNotFoundException.class.isInstance( ex ) ) {
|
|
||||||
if ( DEBUGGING ) {
|
|
||||||
LOG.debug( "Unable to reassemble cached natural-id query result" );
|
|
||||||
}
|
|
||||||
cacheRegion.evict( key );
|
|
||||||
|
|
||||||
// EARLY EXIT !!!!!
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw ex;
|
|
||||||
}
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
catch ( RuntimeException ex ) {
|
||||||
|
if ( isNaturalKeyLookup ) {
|
||||||
|
// potentially perform special handling for natural-id look ups.
|
||||||
|
if ( UnresolvableObjectException.class.isInstance( ex )
|
||||||
|
|| EntityNotFoundException.class.isInstance( ex ) ) {
|
||||||
|
if ( DEBUGGING ) {
|
||||||
|
LOG.debug( "Unable to reassemble cached natural-id query result" );
|
||||||
|
}
|
||||||
|
cacheRegion.evict( key );
|
||||||
|
|
||||||
|
// EARLY EXIT !
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw ex;
|
||||||
}
|
}
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private List getCachedResults(QueryKey key, SessionImplementor session) {
|
private List getCachedResults(QueryKey key, SessionImplementor session) {
|
||||||
|
|
|
@ -62,6 +62,7 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
|
|
||||||
private static final CoreMessageLogger LOG = CoreLogging.messageLogger( DefaultLoadEventListener.class );
|
private static final CoreMessageLogger LOG = CoreLogging.messageLogger( DefaultLoadEventListener.class );
|
||||||
|
|
||||||
|
private static final boolean traceEnabled = LOG.isTraceEnabled();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handle the given load event.
|
* Handle the given load event.
|
||||||
|
@ -70,62 +71,43 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
*
|
*
|
||||||
* @throws HibernateException
|
* @throws HibernateException
|
||||||
*/
|
*/
|
||||||
public void onLoad(LoadEvent event, LoadEventListener.LoadType loadType) throws HibernateException {
|
public void onLoad(final LoadEvent event,
|
||||||
final SessionImplementor source = event.getSession();
|
final LoadEventListener.LoadType loadType) throws HibernateException {
|
||||||
|
|
||||||
EntityPersister persister;
|
final EntityPersister persister = getPersister( event );
|
||||||
if ( event.getInstanceToLoad() != null ) {
|
|
||||||
persister = source.getEntityPersister(
|
|
||||||
null,
|
|
||||||
event.getInstanceToLoad()
|
|
||||||
);
|
|
||||||
//the load() which takes an entity does not pass an entityName
|
|
||||||
event.setEntityClassName( event.getInstanceToLoad().getClass().getName() );
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
persister = source.getFactory().getEntityPersister( event.getEntityClassName() );
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( persister == null ) {
|
if ( persister == null ) {
|
||||||
throw new HibernateException( "Unable to locate persister: " + event.getEntityClassName() );
|
throw new HibernateException( "Unable to locate persister: " + event.getEntityClassName() );
|
||||||
}
|
}
|
||||||
|
|
||||||
final Class idClass = persister.getIdentifierType().getReturnedClass();
|
final Class idClass = persister.getIdentifierType().getReturnedClass();
|
||||||
if ( idClass != null && !idClass.isInstance( event.getEntityId() ) ) {
|
if ( idClass != null && !idClass.isInstance( event.getEntityId() ) )
|
||||||
// we may have the kooky jpa requirement of allowing find-by-id where
|
checkIdClass( persister, event, loadType, idClass );
|
||||||
// "id" is the "simple pk value" of a dependent objects parent. This
|
|
||||||
// is part of its generally goofy "derived identity" "feature"
|
doOnLoad( persister, event, loadType );
|
||||||
if ( persister.getEntityMetamodel().getIdentifierProperty().isEmbedded() ) {
|
}
|
||||||
final EmbeddedComponentType dependentIdType =
|
|
||||||
(EmbeddedComponentType) persister.getEntityMetamodel().getIdentifierProperty().getType();
|
private EntityPersister getPersister( final LoadEvent event ) {
|
||||||
if ( dependentIdType.getSubtypes().length == 1 ) {
|
if ( event.getInstanceToLoad() != null ) {
|
||||||
final Type singleSubType = dependentIdType.getSubtypes()[0];
|
//the load() which takes an entity does not pass an entityName
|
||||||
if ( singleSubType.isEntityType() ) {
|
event.setEntityClassName( event.getInstanceToLoad().getClass().getName() );
|
||||||
final EntityType dependentParentType = (EntityType) singleSubType;
|
return event.getSession().getEntityPersister(
|
||||||
final Type dependentParentIdType = dependentParentType.getIdentifierOrUniqueKeyType( source.getFactory() );
|
null,
|
||||||
if ( dependentParentIdType.getReturnedClass().isInstance( event.getEntityId() ) ) {
|
event.getInstanceToLoad()
|
||||||
// yep that's what we have...
|
|
||||||
loadByDerivedIdentitySimplePkValue(
|
|
||||||
event,
|
|
||||||
loadType,
|
|
||||||
persister,
|
|
||||||
dependentIdType,
|
|
||||||
source.getFactory().getEntityPersister( dependentParentType.getAssociatedEntityName() )
|
|
||||||
);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw new TypeMismatchException(
|
|
||||||
"Provided id of the wrong type for class " + persister.getEntityName() + ". Expected: " + idClass
|
|
||||||
+ ", got " + event.getEntityId().getClass()
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
return event.getSession().getFactory().getEntityPersister( event.getEntityClassName() );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
final EntityKey keyToLoad = source.generateEntityKey( event.getEntityId(), persister );
|
private void doOnLoad(
|
||||||
|
final EntityPersister persister,
|
||||||
|
final LoadEvent event,
|
||||||
|
final LoadEventListener.LoadType loadType) {
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
final EntityKey keyToLoad = event.getSession().generateEntityKey( event.getEntityId(), persister );
|
||||||
if ( loadType.isNakedEntityReturned() ) {
|
if ( loadType.isNakedEntityReturned() ) {
|
||||||
//do not return a proxy!
|
//do not return a proxy!
|
||||||
//(this option indicates we are initializing a proxy)
|
//(this option indicates we are initializing a proxy)
|
||||||
|
@ -137,7 +119,7 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
event.setResult( proxyOrLoad( event, persister, keyToLoad, loadType ) );
|
event.setResult( proxyOrLoad( event, persister, keyToLoad, loadType ) );
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
event.setResult( lockAndLoad( event, persister, keyToLoad, loadType, source ) );
|
event.setResult( lockAndLoad( event, persister, keyToLoad, loadType, event.getSession() ) );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -147,6 +129,42 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void checkIdClass(
|
||||||
|
final EntityPersister persister,
|
||||||
|
final LoadEvent event,
|
||||||
|
final LoadEventListener.LoadType loadType,
|
||||||
|
final Class idClass) {
|
||||||
|
// we may have the kooky jpa requirement of allowing find-by-id where
|
||||||
|
// "id" is the "simple pk value" of a dependent objects parent. This
|
||||||
|
// is part of its generally goofy "derived identity" "feature"
|
||||||
|
if ( persister.getEntityMetamodel().getIdentifierProperty().isEmbedded() ) {
|
||||||
|
final EmbeddedComponentType dependentIdType =
|
||||||
|
(EmbeddedComponentType) persister.getEntityMetamodel().getIdentifierProperty().getType();
|
||||||
|
if ( dependentIdType.getSubtypes().length == 1 ) {
|
||||||
|
final Type singleSubType = dependentIdType.getSubtypes()[0];
|
||||||
|
if ( singleSubType.isEntityType() ) {
|
||||||
|
final EntityType dependentParentType = (EntityType) singleSubType;
|
||||||
|
final Type dependentParentIdType = dependentParentType.getIdentifierOrUniqueKeyType( event.getSession().getFactory() );
|
||||||
|
if ( dependentParentIdType.getReturnedClass().isInstance( event.getEntityId() ) ) {
|
||||||
|
// yep that's what we have...
|
||||||
|
loadByDerivedIdentitySimplePkValue(
|
||||||
|
event,
|
||||||
|
loadType,
|
||||||
|
persister,
|
||||||
|
dependentIdType,
|
||||||
|
event.getSession().getFactory().getEntityPersister( dependentParentType.getAssociatedEntityName() )
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new TypeMismatchException(
|
||||||
|
"Provided id of the wrong type for class " + persister.getEntityName() + ". Expected: " + idClass
|
||||||
|
+ ", got " + event.getEntityId().getClass()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
private void loadByDerivedIdentitySimplePkValue(
|
private void loadByDerivedIdentitySimplePkValue(
|
||||||
LoadEvent event,
|
LoadEvent event,
|
||||||
LoadEventListener.LoadType options,
|
LoadEventListener.LoadType options,
|
||||||
|
@ -176,7 +194,7 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
*
|
*
|
||||||
* @throws HibernateException
|
* @throws HibernateException
|
||||||
*/
|
*/
|
||||||
protected Object load(
|
private Object load(
|
||||||
final LoadEvent event,
|
final LoadEvent event,
|
||||||
final EntityPersister persister,
|
final EntityPersister persister,
|
||||||
final EntityKey keyToLoad,
|
final EntityKey keyToLoad,
|
||||||
|
@ -196,20 +214,17 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
persister.setIdentifier( event.getInstanceToLoad(), event.getEntityId(), event.getSession() );
|
persister.setIdentifier( event.getInstanceToLoad(), event.getEntityId(), event.getSession() );
|
||||||
}
|
}
|
||||||
|
|
||||||
Object entity = doLoad( event, persister, keyToLoad, options );
|
final Object entity = doLoad( event, persister, keyToLoad, options );
|
||||||
|
|
||||||
boolean isOptionalInstance = event.getInstanceToLoad() != null;
|
boolean isOptionalInstance = event.getInstanceToLoad() != null;
|
||||||
|
|
||||||
if ( !options.isAllowNulls() || isOptionalInstance ) {
|
if ( entity == null && ( !options.isAllowNulls() || isOptionalInstance ) ) {
|
||||||
if ( entity == null ) {
|
event.getSession()
|
||||||
event.getSession()
|
.getFactory()
|
||||||
.getFactory()
|
.getEntityNotFoundDelegate()
|
||||||
.getEntityNotFoundDelegate()
|
.handleEntityNotFound( event.getEntityClassName(), event.getEntityId() );
|
||||||
.handleEntityNotFound( event.getEntityClassName(), event.getEntityId() );
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
else if ( isOptionalInstance && entity != event.getInstanceToLoad() ) {
|
||||||
if ( isOptionalInstance && entity != event.getInstanceToLoad() ) {
|
|
||||||
throw new NonUniqueObjectException( event.getEntityId(), event.getEntityClassName() );
|
throw new NonUniqueObjectException( event.getEntityId(), event.getEntityClassName() );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -227,13 +242,13 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
*
|
*
|
||||||
* @return The result of the proxy/load operation.
|
* @return The result of the proxy/load operation.
|
||||||
*/
|
*/
|
||||||
protected Object proxyOrLoad(
|
private Object proxyOrLoad(
|
||||||
final LoadEvent event,
|
final LoadEvent event,
|
||||||
final EntityPersister persister,
|
final EntityPersister persister,
|
||||||
final EntityKey keyToLoad,
|
final EntityKey keyToLoad,
|
||||||
final LoadEventListener.LoadType options) {
|
final LoadEventListener.LoadType options) {
|
||||||
|
|
||||||
if ( LOG.isTraceEnabled() ) {
|
if ( traceEnabled ) {
|
||||||
LOG.tracev(
|
LOG.tracev(
|
||||||
"Loading entity: {0}",
|
"Loading entity: {0}",
|
||||||
MessageHelper.infoString( persister, event.getEntityId(), event.getSession().getFactory() )
|
MessageHelper.infoString( persister, event.getEntityId(), event.getSession().getFactory() )
|
||||||
|
@ -281,7 +296,9 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
final LoadEventListener.LoadType options,
|
final LoadEventListener.LoadType options,
|
||||||
final PersistenceContext persistenceContext,
|
final PersistenceContext persistenceContext,
|
||||||
final Object proxy) {
|
final Object proxy) {
|
||||||
LOG.trace( "Entity proxy found in session cache" );
|
if ( traceEnabled ) {
|
||||||
|
LOG.trace( "Entity proxy found in session cache" );
|
||||||
|
}
|
||||||
LazyInitializer li = ( (HibernateProxy) proxy ).getHibernateLazyInitializer();
|
LazyInitializer li = ( (HibernateProxy) proxy ).getHibernateLazyInitializer();
|
||||||
if ( li.isUnwrap() ) {
|
if ( li.isUnwrap() ) {
|
||||||
return li.getImplementation();
|
return li.getImplementation();
|
||||||
|
@ -321,7 +338,9 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
Object existing = persistenceContext.getEntity( keyToLoad );
|
Object existing = persistenceContext.getEntity( keyToLoad );
|
||||||
if ( existing != null ) {
|
if ( existing != null ) {
|
||||||
// return existing object or initialized proxy (unless deleted)
|
// return existing object or initialized proxy (unless deleted)
|
||||||
LOG.trace( "Entity found in session cache" );
|
if ( traceEnabled ) {
|
||||||
|
LOG.trace( "Entity found in session cache" );
|
||||||
|
}
|
||||||
if ( options.isCheckDeleted() ) {
|
if ( options.isCheckDeleted() ) {
|
||||||
EntityEntry entry = persistenceContext.getEntry( existing );
|
EntityEntry entry = persistenceContext.getEntry( existing );
|
||||||
Status status = entry.getStatus();
|
Status status = entry.getStatus();
|
||||||
|
@ -331,7 +350,9 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
}
|
}
|
||||||
return existing;
|
return existing;
|
||||||
}
|
}
|
||||||
LOG.trace( "Creating new proxy for entity" );
|
if ( traceEnabled ) {
|
||||||
|
LOG.trace( "Creating new proxy for entity" );
|
||||||
|
}
|
||||||
// return new uninitialized proxy
|
// return new uninitialized proxy
|
||||||
Object proxy = persister.createProxy( event.getEntityId(), event.getSession() );
|
Object proxy = persister.createProxy( event.getEntityId(), event.getSession() );
|
||||||
persistenceContext.getBatchFetchQueue().addBatchLoadableEntityKey( keyToLoad );
|
persistenceContext.getBatchFetchQueue().addBatchLoadableEntityKey( keyToLoad );
|
||||||
|
@ -353,7 +374,7 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
*
|
*
|
||||||
* @throws HibernateException
|
* @throws HibernateException
|
||||||
*/
|
*/
|
||||||
protected Object lockAndLoad(
|
private Object lockAndLoad(
|
||||||
final LoadEvent event,
|
final LoadEvent event,
|
||||||
final EntityPersister persister,
|
final EntityPersister persister,
|
||||||
final EntityKey keyToLoad,
|
final EntityKey keyToLoad,
|
||||||
|
@ -402,13 +423,12 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
*
|
*
|
||||||
* @return The loaded entity, or null.
|
* @return The loaded entity, or null.
|
||||||
*/
|
*/
|
||||||
protected Object doLoad(
|
private Object doLoad(
|
||||||
final LoadEvent event,
|
final LoadEvent event,
|
||||||
final EntityPersister persister,
|
final EntityPersister persister,
|
||||||
final EntityKey keyToLoad,
|
final EntityKey keyToLoad,
|
||||||
final LoadEventListener.LoadType options) {
|
final LoadEventListener.LoadType options) {
|
||||||
|
|
||||||
final boolean traceEnabled = LOG.isTraceEnabled();
|
|
||||||
if ( traceEnabled ) {
|
if ( traceEnabled ) {
|
||||||
LOG.tracev(
|
LOG.tracev(
|
||||||
"Attempting to resolve: {0}",
|
"Attempting to resolve: {0}",
|
||||||
|
@ -437,7 +457,7 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
return entity;
|
return entity;
|
||||||
}
|
}
|
||||||
|
|
||||||
entity = loadFromSecondLevelCache( event, persister, options, keyToLoad );
|
entity = loadFromSecondLevelCache( event, persister, keyToLoad );
|
||||||
if ( entity != null ) {
|
if ( entity != null ) {
|
||||||
if ( traceEnabled ) {
|
if ( traceEnabled ) {
|
||||||
LOG.tracev(
|
LOG.tracev(
|
||||||
|
@ -453,7 +473,7 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
MessageHelper.infoString( persister, event.getEntityId(), event.getSession().getFactory() )
|
MessageHelper.infoString( persister, event.getEntityId(), event.getSession().getFactory() )
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
entity = loadFromDatasource( event, persister, keyToLoad, options );
|
entity = loadFromDatasource( event, persister );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( entity != null && persister.hasNaturalIdentifier() ) {
|
if ( entity != null && persister.hasNaturalIdentifier() ) {
|
||||||
|
@ -477,26 +497,21 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
*
|
*
|
||||||
* @param event The load event
|
* @param event The load event
|
||||||
* @param persister The persister for the entity being requested for load
|
* @param persister The persister for the entity being requested for load
|
||||||
* @param keyToLoad The EntityKey representing the entity to be loaded.
|
|
||||||
* @param options The load options.
|
|
||||||
*
|
*
|
||||||
* @return The object loaded from the datasource, or null if not found.
|
* @return The object loaded from the datasource, or null if not found.
|
||||||
*/
|
*/
|
||||||
protected Object loadFromDatasource(
|
private Object loadFromDatasource(
|
||||||
final LoadEvent event,
|
final LoadEvent event,
|
||||||
final EntityPersister persister,
|
final EntityPersister persister) {
|
||||||
final EntityKey keyToLoad,
|
|
||||||
final LoadEventListener.LoadType options) {
|
|
||||||
final SessionImplementor source = event.getSession();
|
|
||||||
Object entity = persister.load(
|
Object entity = persister.load(
|
||||||
event.getEntityId(),
|
event.getEntityId(),
|
||||||
event.getInstanceToLoad(),
|
event.getInstanceToLoad(),
|
||||||
event.getLockOptions(),
|
event.getLockOptions(),
|
||||||
source
|
event.getSession()
|
||||||
);
|
);
|
||||||
|
|
||||||
if ( event.isAssociationFetch() && source.getFactory().getStatistics().isStatisticsEnabled() ) {
|
if ( event.isAssociationFetch() && event.getSession().getFactory().getStatistics().isStatisticsEnabled() ) {
|
||||||
source.getFactory().getStatisticsImplementor().fetchEntity( event.getEntityClassName() );
|
event.getSession().getFactory().getStatisticsImplementor().fetchEntity( event.getEntityClassName() );
|
||||||
}
|
}
|
||||||
|
|
||||||
return entity;
|
return entity;
|
||||||
|
@ -557,15 +572,13 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
*
|
*
|
||||||
* @param event The load event
|
* @param event The load event
|
||||||
* @param persister The persister for the entity being requested for load
|
* @param persister The persister for the entity being requested for load
|
||||||
* @param options The load options.
|
|
||||||
*
|
*
|
||||||
* @return The entity from the second-level cache, or null.
|
* @return The entity from the second-level cache, or null.
|
||||||
*/
|
*/
|
||||||
protected Object loadFromSecondLevelCache(
|
private Object loadFromSecondLevelCache(
|
||||||
final LoadEvent event,
|
final LoadEvent event,
|
||||||
final EntityPersister persister,
|
final EntityPersister persister,
|
||||||
final LoadEventListener.LoadType options,
|
final EntityKey entityKey) {
|
||||||
EntityKey entityKey) {
|
|
||||||
|
|
||||||
final SessionImplementor source = event.getSession();
|
final SessionImplementor source = event.getSession();
|
||||||
final boolean useCache = persister.hasCache()
|
final boolean useCache = persister.hasCache()
|
||||||
|
@ -577,36 +590,24 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
final SessionFactoryImplementor factory = source.getFactory();
|
final Object ce = getFromSharedCache( event, persister, source );
|
||||||
final EntityRegionAccessStrategy cache = persister.getCacheAccessStrategy();
|
|
||||||
final Object ck = cache.generateCacheKey(
|
|
||||||
event.getEntityId(),
|
|
||||||
persister,
|
|
||||||
factory,
|
|
||||||
source.getTenantIdentifier()
|
|
||||||
);
|
|
||||||
|
|
||||||
final Object ce = CacheHelper.fromSharedCache( source, ck, persister.getCacheAccessStrategy() );
|
|
||||||
if ( factory.getStatistics().isStatisticsEnabled() ) {
|
|
||||||
if ( ce == null ) {
|
|
||||||
factory.getStatisticsImplementor().secondLevelCacheMiss(
|
|
||||||
cache.getRegion().getName()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
factory.getStatisticsImplementor().secondLevelCacheHit(
|
|
||||||
cache.getRegion().getName()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( ce == null ) {
|
if ( ce == null ) {
|
||||||
// nothing was found in cache
|
// nothing was found in cache
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
CacheEntry entry = (CacheEntry) persister.getCacheEntryStructure().destructure( ce, factory );
|
return processCachedEntry( event, persister, ce, source, entityKey );
|
||||||
final Object entity;
|
}
|
||||||
|
|
||||||
|
private Object processCachedEntry(
|
||||||
|
final LoadEvent event,
|
||||||
|
final EntityPersister persister,
|
||||||
|
final Object ce,
|
||||||
|
final SessionImplementor source,
|
||||||
|
final EntityKey entityKey) {
|
||||||
|
|
||||||
|
CacheEntry entry = (CacheEntry) persister.getCacheEntryStructure().destructure( ce, source.getFactory() );
|
||||||
if(entry.isReferenceEntry()) {
|
if(entry.isReferenceEntry()) {
|
||||||
if( event.getInstanceToLoad() != null ) {
|
if( event.getInstanceToLoad() != null ) {
|
||||||
throw new HibernateException(
|
throw new HibernateException(
|
||||||
|
@ -614,23 +615,51 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
"is storing references: "+ event.getEntityId()));
|
"is storing references: "+ event.getEntityId()));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
entity = convertCacheReferenceEntryToEntity( (ReferenceCacheEntryImpl) entry,
|
return convertCacheReferenceEntryToEntity( (ReferenceCacheEntryImpl) entry,
|
||||||
event.getEntityId(), persister, event.getSession(), entityKey, event );
|
event.getEntityId(), persister, event.getSession(), entityKey, event );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
entity = convertCacheEntryToEntity( entry, event.getEntityId(), persister, event, entityKey );
|
Object entity = convertCacheEntryToEntity( entry, event.getEntityId(), persister, event, entityKey );
|
||||||
}
|
|
||||||
|
|
||||||
if ( !persister.isInstance( entity ) ) {
|
if ( !persister.isInstance( entity ) ) {
|
||||||
throw new WrongClassException(
|
throw new WrongClassException(
|
||||||
"loaded object was of wrong class " + entity.getClass(),
|
"loaded object was of wrong class " + entity.getClass(),
|
||||||
event.getEntityId(),
|
event.getEntityId(),
|
||||||
persister.getEntityName()
|
persister.getEntityName()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return entity;
|
return entity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private Object getFromSharedCache(
|
||||||
|
final LoadEvent event,
|
||||||
|
final EntityPersister persister,
|
||||||
|
SessionImplementor source ) {
|
||||||
|
final EntityRegionAccessStrategy cache = persister.getCacheAccessStrategy();
|
||||||
|
final Object ck = cache.generateCacheKey(
|
||||||
|
event.getEntityId(),
|
||||||
|
persister,
|
||||||
|
source.getFactory(),
|
||||||
|
source.getTenantIdentifier()
|
||||||
|
);
|
||||||
|
|
||||||
|
final Object ce = CacheHelper.fromSharedCache( source, ck, persister.getCacheAccessStrategy() );
|
||||||
|
if ( source.getFactory().getStatistics().isStatisticsEnabled() ) {
|
||||||
|
if ( ce == null ) {
|
||||||
|
source.getFactory().getStatisticsImplementor().secondLevelCacheMiss(
|
||||||
|
cache.getRegion().getName()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
source.getFactory().getStatisticsImplementor().secondLevelCacheHit(
|
||||||
|
cache.getRegion().getName()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ce;
|
||||||
}
|
}
|
||||||
|
|
||||||
private Object convertCacheReferenceEntryToEntity(
|
private Object convertCacheReferenceEntryToEntity(
|
||||||
|
@ -649,11 +678,14 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
else {
|
else {
|
||||||
makeEntityCircularReferenceSafe(referenceCacheEntry, entityId, session, entity, entityKey);
|
makeEntityCircularReferenceSafe(referenceCacheEntry, entityId, session, entity, entityKey);
|
||||||
//PostLoad is needed for EJB3
|
//PostLoad is needed for EJB3
|
||||||
|
//but not for reference cached entities??
|
||||||
|
/*
|
||||||
EventListenerGroup<PostLoadEventListener> evenListenerGroup = getEvenListenerGroup(session);
|
EventListenerGroup<PostLoadEventListener> evenListenerGroup = getEvenListenerGroup(session);
|
||||||
|
|
||||||
if(!evenListenerGroup.isEmpty()) {
|
if(!evenListenerGroup.isEmpty()) {
|
||||||
postLoad(session, evenListenerGroup.listeners(), entity, entityId, persister, loadEvent);
|
postLoad(session, evenListenerGroup.listeners(), entity, entityId, persister, loadEvent);
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
return entity;
|
return entity;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -676,7 +708,7 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
Object entity,
|
Object entity,
|
||||||
EntityKey entityKey) {
|
EntityKey entityKey) {
|
||||||
|
|
||||||
final EntityPersister subclassPersister = referenceCacheEntry.getSubclassPersister();
|
//final EntityPersister subclassPersister = referenceCacheEntry.getSubclassPersister();
|
||||||
// make it circular-reference safe
|
// make it circular-reference safe
|
||||||
final StatefulPersistenceContext statefulPersistenceContext = (StatefulPersistenceContext) session.getPersistenceContext();
|
final StatefulPersistenceContext statefulPersistenceContext = (StatefulPersistenceContext) session.getPersistenceContext();
|
||||||
|
|
||||||
|
@ -690,14 +722,14 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
TwoPhaseLoad.addUninitializedCachedEntity(
|
TwoPhaseLoad.addUninitializedCachedEntity(
|
||||||
entityKey,
|
entityKey,
|
||||||
entity,
|
entity,
|
||||||
subclassPersister,
|
referenceCacheEntry.getSubclassPersister(),
|
||||||
LockMode.NONE,
|
LockMode.NONE,
|
||||||
referenceCacheEntry.getVersion(),
|
referenceCacheEntry.getVersion(),
|
||||||
session
|
session
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
subclassPersister.afterInitialize( entity, session );
|
//subclassPersister.afterInitialize( entity, referenceCacheEntry.areLazyPropertiesUnfetched(), session );
|
||||||
statefulPersistenceContext.initializeNonLazyCollections();
|
statefulPersistenceContext.initializeNonLazyCollections();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -712,7 +744,7 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
final SessionFactoryImplementor factory = session.getFactory();
|
final SessionFactoryImplementor factory = session.getFactory();
|
||||||
final EntityPersister subclassPersister;
|
final EntityPersister subclassPersister;
|
||||||
|
|
||||||
if ( LOG.isTraceEnabled() ) {
|
if ( traceEnabled ) {
|
||||||
LOG.tracef(
|
LOG.tracef(
|
||||||
"Converting second-level cache entry [%s] into entity : %s",
|
"Converting second-level cache entry [%s] into entity : %s",
|
||||||
entry,
|
entry,
|
||||||
|
@ -808,7 +840,7 @@ public class DefaultLoadEventListener extends AbstractLockUpgradeEventListener i
|
||||||
final EventSource session = event.getSession();
|
final EventSource session = event.getSession();
|
||||||
final SessionFactoryImplementor factory = session.getFactory();
|
final SessionFactoryImplementor factory = session.getFactory();
|
||||||
|
|
||||||
if ( LOG.isTraceEnabled() ) {
|
if ( traceEnabled ) {
|
||||||
LOG.tracev(
|
LOG.tracev(
|
||||||
"Assembling entity from second-level cache: {0}",
|
"Assembling entity from second-level cache: {0}",
|
||||||
MessageHelper.infoString( persister, id, factory )
|
MessageHelper.infoString( persister, id, factory )
|
||||||
|
|
|
@ -1009,7 +1009,7 @@ public final class SessionImpl extends AbstractSessionImpl implements EventSourc
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Object internalLoad(String entityName, Serializable id, boolean eager, boolean nullable)
|
public final Object internalLoad(String entityName, Serializable id, boolean eager, boolean nullable)
|
||||||
throws HibernateException {
|
throws HibernateException {
|
||||||
// todo : remove
|
// todo : remove
|
||||||
LoadEventListener.LoadType type = nullable
|
LoadEventListener.LoadType type = nullable
|
||||||
|
|
Loading…
Reference in New Issue