摘要:使用Spring Data JPA获取的对象,其属性变更后自动更新数据库问题排查与解决方案。
§问题描述
使用继承了JpaRepository的Dao从数据库中获取到某个对象,然后操作这个对象的set属性时,新值直接更新到了数据库。例如,UserDao继承了JpaRepository,从数据库查询出一个User类实例user,当对user执行
user.setAddress("楼兰胡杨门牌号是--" + UUID.randomUUID().toString());
新值会直接更新到数据库。
§问题分析
后来专门重现了这个问题,发现事务结束时会把属性发生改变的user对象更新到数据库,在控制台打印的日志中也可以看见相关update sql语句。百思不得其解,为什么从数据库中获取到的对象属性发生变更的时候,数据库就自动执行更新操作呢?原来是JPA的自带特性在作怪:Hibernate对象有四种状态,【持久态对象的属性被修改后,在数据库事务提交的时候,会更新数据库。】
§问题解决
因持久层框架JPA自身的机制,会在事务提交后将持久状态的对象自动更新到数据库,所以,为了避免自动更新,我们可以创建一个对象,设置好属性后再调用save方法更新数据。
自动更新的好处:如果某个持久状态的对象的确需要更新属性,那么,使用JPA就可以不用手动调用save方法,事务提交后自动更新到数据库,神奇吧?
§阐述实体对象的四种状态以及转换关系
Spring Data JPA 可以理解为 是对JPA 规范的二次封装和抽象,底层还是使用了 Hibernate 的 JPA 技术实现。因此,还是要不辞劳苦地翻阅Hibernate文档。最新的Hibernate文档中为Hibernate对象定义了四种状态(原来是三种状态,面试的时候基本上问的也是三种状态),分别是:瞬时态(new, or transient)、持久态(managed, or persistent)、游状态(detached)和移除态(removed,以前Hibernate文档中定义的三种状态中没有移除态),如下图所示,就以前的Hibernate文档中移除态被视为是瞬时态。
- 瞬时态:当new一个实体对象后,这个对象处于瞬时态,即这个对象只是一个保存临时数据的内存区域,如果没有变量引用这个对象,则会被JVM的垃圾回收机制回收。这个对象所保存的数据与数据库没有任何关系,除非通过Session的save()、saveOrUpdate()、persist()、merge()方法把瞬时态对象与数据库关联,并把数据插入或者更新到数据库,这个对象才转换为持久态对象。
- 持久态:持久态对象的实例在数据库中有对应的记录,并拥有一个持久化标识(ID)。对持久态对象进行delete操作后,数据库中对应的记录将被删除,那么持久态对象与数据库记录不再存在对应关系,持久态对象变成移除态(可以视为瞬时态)。持久态对象被修改变更后,不会马上同步到数据库,直到数据库事务提交。
- 游离态:当Session进行了close()、clear()、evict()或flush()后,实体对象从持久态变成游离态,对象虽然拥有持久和与数据库对应记录一致的标识值,但是因为对象已经从会话中清除掉,对象不在持久化管理之内,所以处于游离态(也叫脱管态)。游离态的对象与临时状态对象是十分相似的,只是它还含有持久化标识。
§小结
以上就是这篇文章关于Spring Data JPA中对象属性改变自动更新数据库的全部内容了,希望本文对大家的学习或者工作能带来一定的帮助,如有疑问请留言交流。Wiener在此祝各位生活愉快!工作顺利!