通常下面的场景会使用update()或saveOrUpdate():
· 程序在第一个session中加载对象
· 该对象被传递到表现层
· 对象发生了一些改动
· 该对象被返回到业务逻辑层
· 程序调用第二个session的update()方法持久这些改动
saveOrUpdate()做下面的事:
· 如果对象已经在本session中持久化了,不做任何事
· 如果另一个与本session关联的对象拥有相同的持久化标识(identifier),抛出一个异常
· 如果对象没有持久化标识(identifier)属性,对其调用save()
· 如果对象的持久标识(identifier)表明其是一个新实例化的对象,对其调用save()
· 如果对象是附带版本信息的(通过
· 否则update() 这个对象
merge()可非常不同:
· 如果session中存在相同持久化标识(identifier)的实例,用用户给出的对象的状态覆盖旧有的持久实例
· 如果session没有相应的持久实例,则尝试从数据库中加载,或创建新的持久化实例, 然后用用户给出的对象的状态覆盖持久实例
· 最后返回该持久实例
· 用户给出的这个对象没有被关联到session上,它依旧是脱管的
merge 和saveOrUpdate 除了 update / save的动作外, 在对象关联关系不存在的时候,还会产生delete的动作, 比如Role和Permission对象有个一对多的关联,当把role对象的Permission 列表设置成空列表或remove部分Permission后, Hibernate会产生相应的delete语句删除数据库中Permission相应数据。
merge和saveOrUpdate的区别在于, 在session中如果没有相应的持久化实例,merge总是先会到数据库中查询对象, 然后通过对比来决定如何更新。
saveOrUpdate不会去查询数据库进行比对, 所以, 他总是执行save 或 update 或 delete语句。
比如, 上面Role和Permission对象一对多的关联,如果用户给出的role.perms 发生了变化, merge(role)操作会先查询数据库加载持久化实例, 然后比较用户给出的role.perms和持久化实例中的perms, 根据实际情况决定insert/update/delete操作。
但是如果是用saveOrUpdate(role), 那么Hibernate 不管三七二十一, 会先在数据库中 update/save role, 然后在Permisssion表中delete所有与这个role相关的permission, 然后再将role.perms中的Permission信息一一插入数据库中,可以看出,这个过程缺少了merge操作的比对过程, 产生的update、insert和delete语句也会比较多。
Hibernate三种状态的区分,以及save,update,saveOrUpdate,merge等的使用
Hibernate的对象有3种状态,分别为:瞬时态(Transient)、 持久态(Persistent)、脱管态(Detached)。处于持久态的对象也称为PO(Persistence Object),瞬时对象和脱管对象也称为VO(Value Object)。 eg. Person person = new Person("xxx", "xx"); 如果没有变量对该对象进行引用,它将被java虚拟机回收。 瞬时对象在内存孤立存在,它是携带信息的载体,不和数据库的数据有任何关联关系,在Hibernate中,可通过session的save()或 saveOrUpdate()方法将瞬时对象与数据库相关联,并将数据对应的插入数据库中,此时该瞬时对象转变成持久化对象。 持久态 当一个session执行close()或clear()、evict()之后,持久对象变成脱管对象,此时持久对象会变成脱管对象,此时该对象虽然具有数据库识别值,但它已不在HIbernate持久层的管理之下。 持久对象具有如下特点: 1. 和session实例关联; 2. 在数据库中有与之关联的记录。 脱管态 脱管对象拥有数据库的识别值,可通过update()、saveOrUpdate()等方法,转变成持久对象。 脱管对象具有如下特点: 1. 本质上与瞬时对象相同,在没有任何变量引用它时,JVM会在适当的时候将它回收; 2. 比瞬时对象多了一个数据库记录标识值。
hibernate的各种保存方式的区分(save,persist,update,saveOrUpdte,merge,flush,lock)及 对象的三种状态
save()和persist()将会引发SQL的INSERT,delete()会引发SQLDELETE, 三、update 和saveOrUpdate区别 saveOrUpdate(po)做下面的事: 四、persist和save区别 2,save, 把一个瞬态的实例持久化标识符,及时的产生,它要返回标识符,所以它会立即执行Sql insert 五、saveOrUpdate,merge和update区别 六、flush和update区别 七、lock和update区别 |