16

我得到了好的、旧的和可怕TransientObjectException的,而且,正如在这种情况下经常发生的那样,我在定位代码中什么样的细微错误导致问题时遇到了问题。

我的问题是:有没有办法获取当前 Hibernate 会话中的每个对象的列表?

当我得到这个问题的答案时,我可能已经解决了当前的问题,但是,无论如何,能够列出会话中的所有内容在下次发生这种情况时会有很大帮助。

4

2 回答 2

19

Hibernate 不会向公众公开其内部结构,因此您不会在公共 API 中找到要搜索的内容。但是,您可以在 Hibernate 接口的实现类中找到答案:此方法(取自http://code.google.com/p/bo2/source/browse/trunk/Bo2ImplHibernate/main/gr/interamerican/bo2/ impl/open/hibernate/HibernateBo2Utils.java ) 将判断会话中是否存在对象:

public static Object getFromSession
        (Serializable identifier, Class<?> clazz, Session s) {              
    String entityName = clazz.getName();
    if(identifier == null) {
       return null;
    }      
    SessionImplementor sessionImpl = (SessionImplementor) s;
    EntityPersister entityPersister = sessionImpl.getFactory().getEntityPersister(entityName);
    PersistenceContext persistenceContext = sessionImpl.getPersistenceContext();
    EntityKey entityKey = new EntityKey(identifier, entityPersister, EntityMode.POJO);
    Object entity = persistenceContext.getEntity(entityKey);
    return entity;
    }

如果再深入一点,您会看到 PersistenceContext 的唯一实现是 org.hibernate.engine.StatefulPersistenceContext。此类具有以下集合:

// Loaded entity instances, by EntityKey
private Map entitiesByKey;

// Loaded entity instances, by EntityUniqueKey
private Map entitiesByUniqueKey;

// Identity map of EntityEntry instances, by the entity instance
private Map entityEntries;

// Entity proxies, by EntityKey
private Map proxiesByKey;

// Snapshots of current database state for entities
// that have *not* been loaded
private Map entitySnapshotsByKey;

// Identity map of array holder ArrayHolder instances, by the array instance
private Map arrayHolders;

// Identity map of CollectionEntry instances, by the collection wrapper
private Map collectionEntries;

// Collection wrappers, by the CollectionKey
private Map collectionsByKey; //key=CollectionKey, value=PersistentCollection

// Set of EntityKeys of deleted objects
private HashSet nullifiableEntityKeys;

// properties that we have tried to load, and not found in the database
private HashSet nullAssociations;

// A list of collection wrappers that were instantiating during result set
// processing, that we will need to initialize at the end of the query
private List nonlazyCollections;

// A container for collections we load up when the owning entity is not
// yet loaded ... for now, this is purely transient!
private Map unownedCollections;

// Parent entities cache by their child for cascading
// May be empty or not contains all relation 
private Map parentsByChild;

因此,您需要做的是将 PersistenceContext 转换为 StatefulPersistenceContext,然后使用反射来获取您想要的私有集合,然后对其进行迭代。

我强烈建议您仅在调试代码时这样做。这不是公共 API,它可能会因 Hibernate 的较新版本而中断。

于 2013-05-09T12:26:26.650 回答
13

发现@nakosspy 帖子非常有用。受他的帖子启发,我添加了这个非常简单的实用方法,可以输出 Hibernate Session 的内容。

正如 nakosspy 所说,这仅用于调试目的,因为它是一个 HACK。

    public static void dumpHibernateSession(Session s) {
    try {
        SessionImplementor sessionImpl = (SessionImplementor) s;
        PersistenceContext persistenceContext = sessionImpl.getPersistenceContext();
        Field entityEntriesField = StatefulPersistenceContext.class.getDeclaredField("entityEntries");
        entityEntriesField.setAccessible(true);
        IdentityMap map = (IdentityMap) entityEntriesField.get(persistenceContext);
        log.info(map);
    } catch (Exception e)
    {
        log.error(e);
    }

}
于 2014-07-04T08:54:46.010 回答