今天作hibernate开发的时候遇到这样一个错误:
a different object with the same identifier value was already associated with the session
(不一样对象具备相同的标识符值已经与会话关联起来)
牵涉出了Hibernate中几个易混淆方法的区别:
首先要介绍下Hibernate中的三种状态
Hibernate的对象有3种状态,分别为:瞬时态(Transient)、持久态(Persistent)、脱管态(Detached)。处于持久态 的对象也称为PO(Persistence Object),瞬时对象和脱管对象也称为VO(Value Object)。java
瞬时态由new命令开辟内存空间的java对象,eg. Person person = new Person("xxx", "xx");如
果没有变量对该对象进行引用,它将被java虚拟机回收。瞬时对象在内存孤立存在,它是携带信息的载体,不和数据库的数据有任何关联关系,在
Hibernate中,可经过session的save()或 saveOrUpdate()方法将瞬时对象与数据库相关联,并将数据对应的插入数据库
中,此时该瞬时对象转变成持久化对象。数据库
持久态处于该状态的对象在数据库中具备对应的记录,并拥有一个持久化标识。若是是
用hibernate的delete()方法,对应的持久对象就变成瞬时对象, 因数据库中的对应数据已被删除,该对象再也不与数据库的记录关联。当一个
session执行close()或clear()、evict()以后,持久对象变成脱管对象,此时持久对象会变成脱管对象,此时该对象虽然具备 数据
库识别值,但它已不在HIbernate持久层的管理之下。 session
持久对象具备以下特色:ide
1.、和session实例关联;
2.、在数据库中有与之关联的记录。spa
脱管态 hibernate
脱管态当与某持久对象关联的session被关闭后,该持久对象转变为脱管对象。当脱管对象被从新关联到session上时,并再次转变成持久对象。脱管对象拥有数据库的识别值,可经过update()、saveOrUpdate()等方法,转变成持久对象。对象
脱管对象拥有数据库的识别值,可经过update()、saveOrUpdate()等方法,转变成持久对象。内存
脱管对象具备以下特色:ci
1. 本质上与瞬时对象相同,在没有任何变量引用它时,JVM会在适当的时候将它回收;开发
2. 比瞬时对象多了一个数据库记录标识值。
下来说下我对merge和saveOrUpdate方法区别的理解:merge方法是把咱们提供的对象转变为托管状态的对象;而saveOrUpdate则是把咱们提供的对象变成一个持久化对象;说的通俗一点就是:saveOrUpdate后的对象会归入session的管理,对象的状态会跟数据库同步,再次查询该对象会直接从session中取,merge后的对 象不会归入session的管理,再次查询该对象仍是会从数据库中取。
因此遇到“a different object with the same identifier value was already associated with the session”这个问题只须要清空session或者将saveOrUpdate方法变为merge方法就能够了。