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状态,后续代码对其的修改,无法合并到数据库中
{
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还是托管状态。
我要举报
如以上回答内容为低俗、色情、不良、暴力、侵权、涉及违法等信息,可以点下面链接进行举报!
点此我要举报以上问答信息
大家都在看
推荐资讯