我得到了好的、旧的和可怕TransientObjectException
的,而且,正如在这种情况下经常发生的那样,我在定位代码中什么样的细微错误导致问题时遇到了问题。
我的问题是:有没有办法获取当前 Hibernate 会话中的每个对象的列表?
当我得到这个问题的答案时,我可能已经解决了当前的问题,但是,无论如何,能够列出会话中的所有内容在下次发生这种情况时会有很大帮助。
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 的较新版本而中断。
发现@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);
}
}