Выселить зависимые коллекции вместе с родительской сущностью - PullRequest
6 голосов
/ 06 марта 2012

Я только что понял, что когда объект выгружается из кэша Hibernate, зависимые коллекции, если они кэшированы, должны быть выселены отдельно .

Для меня это один большой WTF:

  • действительно легко забыть выселить коллекцию (например, когда новая добавлена ​​в отображение объекта);
  • код для выселения зависимых коллекций уродлив и громоздок, например

    MyClass myObject = ...;getHibernateTemplate().evict(myObject);Cache cache = getHibernateTemplate().getSessionFactory().getCache();cache.evictCollection("my.package.MyClass.myCollection1, id);...cache.evictCollection("my.package.MyClass.myCollectionN, id);

Совершенно очевидно, что если родительский объект изменился, нет смысла хранить его коллекции, так как они, скорее всего, все равно будут получены из этого родителя.

Я что-то здесь упускаю?Неужели нет способа сбрасывать объект вместе со всеми его дочерними объектами без написания всего этого кода вручную?

Ответы [ 2 ]

4 голосов
/ 30 августа 2013

Это старая проблема . Есть способ подключиться к спящему режиму, чтобы исключить кэши коллекций при вставке, обновлении или удалении упомянутой сущности коллекций. У меня есть исправление для спящего режима . Исправление запланировано для Hibernate 4.3.0.Beta5 и будет активировано свойством:

hibernate.cache.auto_evict_collection_cache=true

Пока это исправление не выпущено, вы можете обойти проблему, чтобы внедрить логику выселения, просто зарегистрировав CollectionCacheInvalidator в SessionFactory и SessionFactoryServiceRegistry самостоятельно.

import javax.persistence.OneToMany;
import java.io.Serializable;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;

import my.own.library.BeanInformationFromClass;
import my.own.library.PropertyInformationFromClass;
import org.apache.commons.lang.StringUtils;
import org.apache.log4j.Logger;

import org.hibernate.engine.spi.SessionFactoryImplementor;
import org.hibernate.event.service.spi.EventListenerRegistry;
import org.hibernate.event.spi.EventSource;
import org.hibernate.event.spi.EventType;
import org.hibernate.event.spi.PostInsertEvent;
import org.hibernate.event.spi.PostInsertEventListener;
import org.hibernate.event.spi.PreDeleteEvent;
import org.hibernate.event.spi.PreDeleteEventListener;
import org.hibernate.event.spi.PreUpdateEvent;
import org.hibernate.event.spi.PreUpdateEventListener;
import org.hibernate.persister.collection.CollectionPersister;
import org.hibernate.persister.entity.EntityPersister;
import org.hibernate.persister.entity.Joinable;
import org.hibernate.service.spi.SessionFactoryServiceRegistry;

/**
 * @author Andreas Berger (latest modification by $Author$)
 * @version $Id$
 * @created 27.08.13 - 17:49
 */
public class CollectionCacheInvalidator
        implements PostInsertEventListener, PreDeleteEventListener, PreUpdateEventListener {

    private static final Logger LOGGER = Logger.getLogger( CollectionCacheInvalidator.class );

    private Map<String, String> mappedByFieldMapping;

    public void integrate(SessionFactoryImplementor sf, SessionFactoryServiceRegistry registry) {
        EventListenerRegistry eventListenerRegistry = registry.getService( EventListenerRegistry.class );
        eventListenerRegistry.appendListeners( EventType.POST_INSERT, this );
        eventListenerRegistry.appendListeners( EventType.PRE_DELETE, this );
        eventListenerRegistry.appendListeners( EventType.PRE_UPDATE, this );

        mappedByFieldMapping = new HashMap<String, String>();

        Map<String, CollectionPersister> persiters = sf.getCollectionPersisters();
        if ( persiters != null ) {
            for ( CollectionPersister collectionPersister : persiters.values() ) {
                if ( !collectionPersister.hasCache() ) {
                    continue;
                }
                if ( !(collectionPersister instanceof Joinable) ) {
                    continue;
                }
                String oneToManyFieldName = collectionPersister.getNodeName();
                EntityPersister ownerEntityPersister = collectionPersister.getOwnerEntityPersister();
                Class ownerClass = ownerEntityPersister.getMappedClass();

                // Logic to get the mappedBy attribute of the OneToMany annotation.
                BeanInformationFromClass bi = new BeanInformationFromClass( ownerClass );
                PropertyInformationFromClass prop = bi.getProperty( oneToManyFieldName );
                OneToMany oneToMany = prop.getAnnotation( OneToMany.class );
                String mappedBy = null;
                if ( oneToMany != null && StringUtils.isNotBlank( oneToMany.mappedBy() ) ) {
                    mappedBy = oneToMany.mappedBy();
                }
                mappedByFieldMapping.put( ((Joinable) collectionPersister).getName(), mappedBy );
            }
        }
    }

    @Override
    public void onPostInsert(PostInsertEvent event) {
        evictCache( event.getEntity(), event.getPersister(), event.getSession(), null );
    }

    @Override
    public boolean onPreDelete(PreDeleteEvent event) {
        evictCache( event.getEntity(), event.getPersister(), event.getSession(), null );
        return false;
    }

    @Override
    public boolean onPreUpdate(PreUpdateEvent event) {
        evictCache( event.getEntity(), event.getPersister(), event.getSession(), event.getOldState() );
        return false;
    }

    private void evictCache(Object entity, EntityPersister persister, EventSource session, Object[] oldState) {
        try {
            SessionFactoryImplementor factory = persister.getFactory();

            Set<String> collectionRoles = factory.getCollectionRolesByEntityParticipant( persister.getEntityName() );
            if ( collectionRoles == null || collectionRoles.isEmpty() ) {
                return;
            }
            for ( String role : collectionRoles ) {
                CollectionPersister collectionPersister = factory.getCollectionPersister( role );
                if ( !collectionPersister.hasCache() ) {
                    continue;
                }
                if ( !(collectionPersister instanceof Joinable) ) {
                    continue;
                }
                String mappedBy = mappedByFieldMapping.get( ((Joinable) collectionPersister).getName() );
                if ( mappedBy != null ) {
                    int i = persister.getEntityMetamodel().getPropertyIndex( mappedBy );
                    Serializable oldId = null;
                    if ( oldState != null ) {
                        oldId = session.getIdentifier( oldState[i] );
                    }
                    Object ref = persister.getPropertyValue( entity, i );
                    Serializable id = null;
                    if ( ref != null ) {
                        id = session.getIdentifier( ref );
                    }
                    if ( id != null && !id.equals( oldId ) ) {
                        evict( id, collectionPersister, session );
                        if ( oldId != null ) {
                            evict( id, collectionPersister, session );
                        }
                    }
                }
                else {
                    LOGGER.debug( "Evict CollectionRegion " + role );
                    collectionPersister.getCacheAccessStrategy().evictAll();
                }
            }
        }
        catch (Exception e) {
            LOGGER.error( "", e );
        }
    }

    private void evict(Serializable id, CollectionPersister collectionPersister, EventSource session) {
        LOGGER.debug( "Evict CollectionRegion " + collectionPersister.getRole() + " for id " + id );
        collectionPersister.getCacheAccessStrategy().evict(
                session.generateCacheKey(
                        id,
                        collectionPersister.getKeyType(),
                        collectionPersister.getRole()
                )
        );
    }
}
0 голосов
/ 27 марта 2012

Это просто кеш.Кеш просто должен уменьшить доступ к базе данных.Когда вы выселяете объект, часто вы не вносите никаких изменений в дочерние объекты, и они просто могут быть загружены из кэша в следующий раз.Также часто случается, что дочерние объекты все еще используются другими родительскими объектами (в этом случае имя 'child' неверно, потому что это отношение n: 1 или m: n).Изгнание детей может вызвать очень странные ошибки в другом месте, где дочерние объекты все еще используются.

Так что, если удастся выселить детей, зависит только от вашего приложения и дизайна базы данных.Таким образом, hibernate по умолчанию не исключает дочерние объекты.

Если вы хотите, чтобы дочерние объекты автоматически выселялись, используйте cascade = "evict" в вашем файле сопоставления.

Более рабатский метод для удалениявсе объекты - закрыть сеанс и открыть новый.Затем все объекты сеанса выселяются.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...