今天做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)。
瞬时态由new命令开辟内存空间的java对象,eg. Person person = new Person("xxx", "xx");如
果没有变量对该对象进行引用,它将被java虚拟机回收。瞬时对象在内存孤立存在,它是携带信息的载体,不和数据库的数据有任何关联关系,在
Hibernate中,可通过session的save()或 saveOrUpdate()方法将瞬时对象与数据库相关联,并将数据对应的插入数据库
中,此时该瞬时对象转变成持久化对象。
持久态处于该状态的对象在数据库中具有对应的记录,并拥有一个持久化标识。如果是
用hibernate的delete()方法,对应的持久对象就变成瞬时对象, 因数据库中的对应数据已被删除,该对象不再与数据库的记录关联。当一个
session执行close()或clear()、evict()之后,持久对象变成脱管对象,此时持久对象会变成脱管对象,此时该对象虽然具有 数据
库识别值,但它已不在HIbernate持久层的管理之下。
持久对象具有如下特点:
1.、和session实例关联;
2.、在数据库中有与之关联的记录。
脱管态
脱管态当与某持久对象关联的session被关闭后,该持久对象转变为脱管对象。当脱管对象被重新关联到session上时,并再次转变成持久对象。脱管对象拥有数据库的识别值,可通过update()、saveOrUpdate()等方法,转变成持久对象。
脱管对象拥有数据库的识别值,可通过update()、saveOrUpdate()等方法,转变成持久对象。
脱管对象具有如下特点:
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方法就可以了。