/*
 * Hibernate, Relational Persistence for Idiomatic Java
 *
 * Copyright (c) 2012, Red Hat Inc. or third-party contributors as
 * indicated by the @author tags or express copyright attribution
 * statements applied by the authors.  All third-party contributions are
 * distributed under license by Red Hat Inc.
 *
 * This copyrighted material is made available to anyone wishing to use, modify,
 * copy, or redistribute it subject to the terms and conditions of the GNU
 * Lesser General Public License, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
 * or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public License
 * for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with this distribution; if not, write to:
 * Free Software Foundation, Inc.
 * 51 Franklin Street, Fifth Floor
 * Boston, MA  02110-1301  USA
 */
package org.hibernate.cache.spi.entry;

import java.io.Serializable;

import org.hibernate.AssertionFailure;
import org.hibernate.HibernateException;
import org.hibernate.Interceptor;
import org.hibernate.engine.spi.SessionImplementor;
import org.hibernate.event.service.spi.EventListenerGroup;
import org.hibernate.event.service.spi.EventListenerRegistry;
import org.hibernate.event.spi.EventSource;
import org.hibernate.event.spi.EventType;
import org.hibernate.event.spi.PreLoadEvent;
import org.hibernate.event.spi.PreLoadEventListener;
import org.hibernate.internal.util.collections.ArrayHelper;
import org.hibernate.persister.entity.EntityPersister;
import org.hibernate.type.TypeHelper;

Standard representation of entity cached data using the "disassembled state".
Author:Steve Ebersole
/** * Standard representation of entity cached data using the "disassembled state". * * @author Steve Ebersole */
public class StandardCacheEntryImpl implements CacheEntry { private final Serializable[] disassembledState; private final String subclass; private final boolean lazyPropertiesAreUnfetched; private final Object version;
Constructs a StandardCacheEntryImpl
Params:
  • state – The extracted state
  • persister – The entity persister
  • unfetched – Are any values present in state unfetched?
  • version – The current version (if versioned)
  • session – The originating session
  • owner – The owner
Throws:
/** * Constructs a StandardCacheEntryImpl * * @param state The extracted state * @param persister The entity persister * @param unfetched Are any values present in state unfetched? * @param version The current version (if versioned) * @param session The originating session * @param owner The owner * * @throws HibernateException Generally indicates a problem performing the dis-assembly. */
public StandardCacheEntryImpl( final Object[] state, final EntityPersister persister, final boolean unfetched, final Object version, final SessionImplementor session, final Object owner) throws HibernateException { // disassembled state gets put in a new array (we write to cache by value!) this.disassembledState = TypeHelper.disassemble( state, persister.getPropertyTypes(), persister.isLazyPropertiesCacheable() ? null : persister.getPropertyLaziness(), session, owner ); subclass = persister.getEntityName(); lazyPropertiesAreUnfetched = unfetched || !persister.isLazyPropertiesCacheable(); this.version = version; } StandardCacheEntryImpl(Serializable[] state, String subclass, boolean unfetched, Object version) { this.disassembledState = state; this.subclass = subclass; this.lazyPropertiesAreUnfetched = unfetched; this.version = version; } @Override public boolean isReferenceEntry() { return false; } @Override public Serializable[] getDisassembledState() { // todo: this was added to support initializing an entity's EntityEntry snapshot during reattach; // this should be refactored to instead expose a method to assemble a EntityEntry based on this // state for return. return disassembledState; } @Override public String getSubclass() { return subclass; } @Override public boolean areLazyPropertiesUnfetched() { return lazyPropertiesAreUnfetched; } @Override public Object getVersion() { return version; }
After assembly, is a copy of the array needed?
Returns:true/false
/** * After assembly, is a copy of the array needed? * * @return true/false */
public boolean isDeepCopyNeeded() { // for now always return true. // todo : See discussion on HHH-7872 return true; }
Assemble the previously disassembled state represented by this entry into the given entity instance. Additionally manages the PreLoadEvent callbacks.
Params:
  • instance – The entity instance
  • id – The entity identifier
  • persister – The entity persister
  • interceptor – (currently unused)
  • session – The session
Throws:
  • HibernateException – Indicates a problem performing assembly or calling the PreLoadEventListeners.
See Also:
Returns:The assembled state
/** * Assemble the previously disassembled state represented by this entry into the given entity instance. * * Additionally manages the PreLoadEvent callbacks. * * @param instance The entity instance * @param id The entity identifier * @param persister The entity persister * @param interceptor (currently unused) * @param session The session * * @return The assembled state * * @throws HibernateException Indicates a problem performing assembly or calling the PreLoadEventListeners. * * @see org.hibernate.type.Type#assemble * @see org.hibernate.type.Type#disassemble */
public Object[] assemble( final Object instance, final Serializable id, final EntityPersister persister, final Interceptor interceptor, final EventSource session) throws HibernateException { if ( !persister.getEntityName().equals( subclass ) ) { throw new AssertionFailure( "Tried to assemble a different subclass instance" ); } //assembled state gets put in a new array (we read from cache by value!) final Object[] assembledProps = TypeHelper.assemble( disassembledState, persister.getPropertyTypes(), session, instance ); //persister.setIdentifier(instance, id); //before calling interceptor, for consistency with normal load //TODO: reuse the PreLoadEvent final PreLoadEvent preLoadEvent = new PreLoadEvent( session ) .setEntity( instance ) .setState( assembledProps ) .setId( id ) .setPersister( persister ); final EventListenerGroup<PreLoadEventListener> listenerGroup = session .getFactory() .getServiceRegistry() .getService( EventListenerRegistry.class ) .getEventListenerGroup( EventType.PRE_LOAD ); for ( PreLoadEventListener listener : listenerGroup.listeners() ) { listener.onPreLoad( preLoadEvent ); } persister.setPropertyValues( instance, assembledProps ); return assembledProps; } @Override public String toString() { return "CacheEntry(" + subclass + ')' + ArrayHelper.toString( disassembledState ); } }