前些天项目中使用hibernate4.1.x,Dao层已经封装好了,有使用sql查询的,hql的等等,我比较懒就是用的myeclipse生成的实体类。正常来说这样挺方便的,而且session也是opensessioninview,事务也是注解的。
问题来了:当我使用hql做查询时,返回一个list集合,当我对这个集合的值进行修改时,数据库对应的也会改变,根本不需要更新或者其他操作
原因:在hibernate中对象有三种状态,如图,当session一个session开启并执行查询时,如果这时的session没有关闭,查询出来的对象为持久化状态,就是当前对象是和数据库同步的,放到了hibernate的缓存中。当这个事务提交时,hibernate会对持久化对象进行脏检查,如果与数据库的数据不一致将会更新这条数据。当session关闭时。该对象的状态为游离状态,垃圾回收之后从缓存中消失。
结论:
1.查询出来的数据属于持久化状态。在事务提交时会进行脏检查。
2.持久化对象和值对象要分开使用。
尝试的解决方式:
1.list集合的addAll()方法,将一个集合全部添加到一个新的集合中(不行)
2.ArrayList的clone()方法,克隆一个集合到新集合中(浅克隆不行)
3.重新创建一个集合,重新new 一个对象,把每一个属性遍历赋到新集合中(可以)
4.在封装的方法中在加一个方法,在查询结束后直接关闭session。不使用currentSession,重新打开一个session。(可以,但是会影响性能)