前俩月在项目里写了个基于hibernate的监听事件,然后往自定义的审计表里插数据的功能。因为之前不需要记录修改前的数据,所以event.getOldState()这个方法一直没用过。
这两天另外一个功能需要用到提交前的记录,结果发现event.getOldState()有时候获得值为null,有时候不是null。
于是有点晕,网上搜了一下,的确也有很多这个情况,但是都没有说原因是什么,有人说用了merge()就有值了,我实际测试完也的确是这样。但是为什么呢?
查问题办法就是跟源代码。
因为我们公司目前在用东软的框架(吐槽一下,还是hibernate3),所以是3.2.3的源码。如果是hibernate4的话,估计应该差不多。另外hibernate4的话,Hibernate Envers这个东西应该很成熟了。
于是打断点,跟踪源码。 DefaultSaveOrUpdateEventListener类中performUpdate方法内
可见第三个参数为null,即我们获得的数值也为空。
并且在方法内有注释说明:// this is a transient object with existing persistent state not loaded by the session.
至此,原因很明显了,我们保存的对象是一个瞬时态的obj,hibernate不是从数据库里把这个对象抓取出来的,是前台框架直接new出来的。
所以前面查到有人说merge之后好用。
结论就是:[b]getOldState()如果要有值,那么一定要在代码中保存持久态的对象。[/b]
这两天另外一个功能需要用到提交前的记录,结果发现event.getOldState()有时候获得值为null,有时候不是null。
于是有点晕,网上搜了一下,的确也有很多这个情况,但是都没有说原因是什么,有人说用了merge()就有值了,我实际测试完也的确是这样。但是为什么呢?
查问题办法就是跟源代码。
因为我们公司目前在用东软的框架(吐槽一下,还是hibernate3),所以是3.2.3的源码。如果是hibernate4的话,估计应该差不多。另外hibernate4的话,Hibernate Envers这个东西应该很成熟了。
于是打断点,跟踪源码。 DefaultSaveOrUpdateEventListener类中performUpdate方法内
source.getPersistenceContext().addEntity(
entity,
Status.MANAGED,
null, //cachedState,
key,
persister.getVersion( entity, source.getEntityMode() ),
LockMode.NONE,
true,
persister,
false,
true
);
可见第三个参数为null,即我们获得的数值也为空。
并且在方法内有注释说明:// this is a transient object with existing persistent state not loaded by the session.
至此,原因很明显了,我们保存的对象是一个瞬时态的obj,hibernate不是从数据库里把这个对象抓取出来的,是前台框架直接new出来的。
所以前面查到有人说merge之后好用。
结论就是:[b]getOldState()如果要有值,那么一定要在代码中保存持久态的对象。[/b]

本文探讨了在使用Hibernate框架时,监听事件与持久化对象状态之间的关系,特别是如何确保getOldState()方法获取到有效值,强调了在代码中保存持久态对象的重要性。
1024

被折叠的 条评论
为什么被折叠?



