永发信息网

session.update和session.merge的区别

答案:2  悬赏:60  手机版
解决时间 2021-04-04 19:45
  • 提问者网友:佞臣
  • 2021-04-04 04:12
session.update和session.merge的区别
最佳答案
  • 五星知识达人网友:纵马山川剑自提
  • 2021-04-04 05:33
public void testMerge()
{

Teacher t = new Teacher();
t.setId(4);
t.setName("力量1");
t.setStudentNum("100");

//t为detached态

session = sf.openSession();

session.beginTransaction();

Teacher t2 = session.get(Teacher.class,4);

session.update(t);//
session.merge(t);//
//System.out.println("id"+t.getId());

//System.out.println("id"+t.getId());
//session.get
//System.out.println(t.getClass());
session.getTransaction().commit();

session.close();
//System.out.println("id"+t.getName());
sf.close();

}

当使用update,因为有一个id为4的persistent态实例,会报错

使用merge时,能成功执行

首先,对于一个处于detatch状态的实体对象来说,要将其中的修改,合并到数据库中,有两种方法,一种方式,是调用update(),另一种方式是调用merge()

当调用update()时,首先要取保目标session中,不包含同样id的实体对象的引用,如果有的话,会抛出异常,完成update()方法后,这个实体对象从detatch状态,转换为persistent状态,在session提交前,后续对其的修改,都会被合并到数据库中。
当调用merge()对象时,无需考虑session中是否已经包含同样id的实体对象,如果session中没有同样id的实体对象,hibernate会通过select语句,从数据库中查询出对应对象,如果数据库中没有对应对象,就新建一个。同时,完成merge()操作后,会返回数据库中对应的persistent状态对象,而原有的,作为参数传入的实体对象,仍然是detatch状态,后续代码对其的修改,无法合并到数据库中
全部回答
  • 1楼网友:三千妖杀
  • 2021-04-04 05:41
当我们使用update的时候,执行完成后,我们提供的对象a的状态变成持久化状态。 但当我们使用merge的时候,执行完成,我们提供的对象a还是脱管状态,hibernate或者new了一个b,或者检索到 一个持久对象b,并把我们提供的对象a的所有的值拷贝到这个b,执行完成后b是持久状态,而我们提供的a还是托管状态。 merge与update区别 注:就因为这2个方法的区别还得我花了太多时间项目迟迟不能做完 ,但是让我解决了,学到了东西了. 这是一段代码 public void updatedata(object obj) { try { tx = this.getsession().begintransaction(); // 执行修改操作 //this.getsession().update(obj); this.getsession().merge(obj); tx.commit(); } catch (exception e) { system.out.println("===修改信息出现异常==="); e.printstacktrace(); tx.rollback(); } } 1. 数据库记录已存在,更改person的name为一个新的name。 merge方法打印出的日志如下: hibernate: select person0_.id as id0_0_, person0_.name as name0_0_ from person person0_ where person0_.id=? hibernate: update person set name=? where id=? update方法打印出的日志如下: hibernate: update person set name=? where id=? 2. 数据库记录已存在,更改person的name和数据库里对应id记录的name一样的值。 merge方法打印出的日志如下: hibernate: select person0_.id as id0_0_, person0_.name as name0_0_ from person person0_ where person0_.id=? 此处相对于第一种情形少了update的动作 update方法打印出的日志如下: hibernate: update person set name=? where id=? 3. 数据库记录不存在时,也就是你传的实体bean的id在数据库没有对应的记录。 merge方法打印出的日志如下: hibernate: select person0_.id as id0_0_, person0_.name as name0_0_ from person person0_ where person0_.id=? hibernate: insert into person (name) values (?) 如果没有对应的记录,merge会把该记录当作新的记录来插入。此处我很疑惑,因为我传得person实体对象里写明了id值的,它为什么还会做插入的动作呢? update方法打印出的日志如下: hibernate: update person set name=? where id=? 2009-11-22 20:59:55,359 error [org.hibernate.jdbc.abstractbatcher] - exception executing batch: org.hibernate.stalestateexception: batch update returned unexpected row count from update [0]; actual row count: 0; expected: 1 以下的内容摘抄自网上: 当我们使用update的时候,执行完成后,我们提供的对象a的状态变成持久化状态。 但当我们使用merge的时候,执行完成,我们提供的对象a还是脱管状态,hibernate或者new了一个b,或者检索到 一个持久对象b,并把我们提供的对象a的所有的值拷贝到这个b,执行完成后b是持久状态,而我们提供的a还是托管状态。
我要举报
如以上回答内容为低俗、色情、不良、暴力、侵权、涉及违法等信息,可以点下面链接进行举报!
点此我要举报以上问答信息
大家都在看
推荐资讯