记一次使用hibernate获取的数据不是实时的问题
前言:当前端把数据库中的某个用户下的事项全部删除时,发现他的事项统计数没有发生改变,查看请求发现他的数据还是之前的并不是数据库现有的数据,并且后台报出ObjectNotFoundException这个异常。
1. 发现问题: 经过各种尝试与排查发现是由于hibernate的load()方法认为该数据在数据库中一定存在,可以放心的使用代理来延迟加载,如果在使用过程中发现了问题,只能抛异常,并返回之前延迟的数据。
2. 分析问题原因: 在hibernate中我们知道如果要从数据库中得到一个对象,通常有两种方式,一种是通过session.get()方法,另一种就是通过session.load()方法,然后其实这两种方法在获得一个实体对象时是有区别的,在查询性能上两者是不同的。
- get() 方法:会先确认id对应的数据是否存在,一定要获取到真实的数据,否则就返回null;
- load() 方法:会先认定id对应的数据一定存在,会先在session的一级缓存中去查找,如果没有找到,会根据lazy属性值来确认是否使用延迟加载,如果lazy=‘true’,就是使用延迟加载,返回该代理对象,等到真正访问到该对象的属性时才去二级缓存中查询,如果没有,再取数据库中查询,如果还没有的话,就抛出org.hibernate.ObjectNotFoundException异常。
3. 解决方案: 当lazy=‘false’ 的时候,就不会使用延迟加载,这个时候load()的访问机制就和get()一样了。
解决样例:
在需要的xxx.hbm.xml里进行修改。