session的save,saveOrUpdate, merge区别!

hibernate的保存

   hibernate对于对象的保存提供了太多的方法,他们之间有很多不同,在这里细说一下,以便区别:

一、预备知识:
     在所有之前,说明一下,对于hibernate,它的对象有三种状态,transient、persistent、detached
下边是常见的翻译办法:
transient:瞬态或者自由态
persistent:持久化状态
detached:脱管状态或者游离态

脱管状态的实例可以通过调用save()、persist()或者saveOrUpdate()方法进行持久化。
持久化实例可以通过调用 delete()变成脱管状态。通过get()或load()方法得到的实例都是持久化状态的。
脱管状态的实例可以通过调用 update()、0saveOrUpdate()、lock()或者replicate()进行持久化。

save()和persist()将会引发SQL的INSERT,delete()会引发SQLDELETE,
而update()或merge()会引发SQLUPDATE。对持久化(persistent)实例的修改在刷新提交的时候会被检测到,
它也会引起SQLUPDATE。saveOrUpdate()或者replicate()会引发SQLINSERT或者UPDATE

二、save 和update区别
       把这一对放在第一位的原因是因为这一对是最常用的。
       save的作用是把一个新的对象保存
       update是把一个脱管状态的对象保存

三,update 和saveOrUpdate区别
      这个是比较好理解的,顾名思义,saveOrUpdate基本上就是合成了save和update
      引用hibernate reference中的一段话来解释他们的使用场合和区别
      通常下面的场景会使用update()或saveOrUpdate(): 
      程序在第一个session中加载对象 
      该对象被传递到表现层 
      对象发生了一些改动 
      该对象被返回到业务逻辑层 
      程序调用第二个session的update()方法持久这些改动

saveOrUpdate()做下面的事: 
       如果对象已经在本session中持久化了,不做任何事 
       如果另一个与本session关联的对象拥有相同的持久化标识(identifier),抛出一个异常 
       如果对象没有持久化标识(identifier)属性,对其调用save() 
      如果对象的持久标识(identifier)表明其是一个新实例化的对象,对其调用save() 
      如果对象是附带版本信息的(通过) 并且版本属性的值表明其 是一个新实例化的对象,save()它。 
否则update() 这个对象

四,persist和save区别
      这个是最迷离的一对,表面上看起来使用哪个都行,在hibernate reference文档中也没有明确的区分他们.
这里给出一个明确的区分。(可以跟进src看一下,虽然实现步骤类似,但是还是有细微的差别)

五,update和merge区别

update():如果你确定在当前session的会话里不存在一个与要进行update操作有相同标识符(主键)的持久化对象,那么调用update()。这句话理解起来有点绕,后面举个例子就明白了。

merge():如果你在任何时候修改了数据都想把数据保存到数据库中,那么就调用merge()。


        例子程序片段:

        有一个Person对象,里面有两个属性,id和name

[java]  view plain  copy
 print ?
  1. Session session = sessionFactory.openSession();   
  2. Transaction tx = session.beginTransaction();   
  3. Person person = (Person) session.get(Person.class"15");  
  4. tx.commit();   
  5. session.clear();  
  6. session.close();  
  7.   
  8. System.out.println(person.getId());  
  9. person.setName("jasper");  
  10. Session session2 = sessionFactory.openSession();   
  11. Transaction tx2 = session2.beginTransaction();   
  12. Person person2 = (Person) session2.get(Person.class"15");  //是不是感觉这句代码有点多余,问题就出在这里  
  13. //session2.update(person);  //抛出 org.hibernate.NonUniqueObjectException  
  14. //session2.saveOrUpdate(person);  //抛出 org.hibernate.NonUniqueObjectException  
  15. session2.merge(person);  //成功执行  
  16. tx2.commit();   
  17. session2.close();  

        当调用update或saveOrUpdate操作时,抛出了异常信息“org.hibernate.NonUniqueObjectException”,不是“独一无二”的对象。为什么会抛出这个错误呢?上面的person2调用了get方法,所以person2是一个持久化对象,当在session还没有关闭的时候对person2做的任何修改,都会保存到数据库中;person要执行更新操作,但由于它是一个脱管对象,那么就需要跟数据库重新取得联系,需要从数据库获取数据,hibernate会查找当前是否关于这个对象的副本,刚好有个person2(但又不知person2是否修改了且保存到了数据库),与其保存这可能没用的数据,还不如直接报错来得直接。这里调用update和saveOrUpdate类似。

        还有update与merge的一点不同是:update是直接发出“sql update”语句,而merge会先发出“sql select”语句,如果没有查询到结果,那么执行save操作,如果查询到相关结果,则判断对应字段是否有发生改变,如果改变了,则发出“sql update”语句,否则什么也不做。这一方面,merge就和saveOrUpdate相同。




你可能感兴趣的:(service,java服务端知识点,后台数据库)