------------吾亦无他,唯手熟尔,谦卑若愚,好学若饥-------------
本篇博客主要叙述Hibernate中的三种状态:临时状态(瞬时状态),持久状态,游离状态
commit和flush三种状态间的使用,commit和flush的区别:
saveOrUpdate和merge三种状态间的使用,saveOrUpdate和merge的区别
前方标注一下,用的是mysql数据库,主键生成策略是increment
一,Hibernate-ORM的对象的三种状态:
* Hibernate对象的三种状态
* 1.临时状态(瞬时态)(临时态,*态)
* 我们通过new关键字创建出一个类的实例对象, 这个对象和hibernate没有任何关系!
* 2.持久状态
* 对象被session管理。就会产生有一个OID(主键标识符)!这个对象和hibernate有关系
* 3.游离状态(托管态)
* 曾经被session管理过!有OID
*和瞬时态的区别在于,是否存在OID!
大话一下:临时状态就是没有关系,就像你和大街上任意一个女孩没有关系,持久状态就是你和她有了联系(男女朋友,结婚),游离状态就是已经分手了(但是有过曾经)
二,一张图揭秘如何进入各个状态的方法:
三,各个状态的小案例:
@Test /*简单写一下三种状态,和根据这个increment主键生成策咯生成记录 * * ---------------------------------- Hibernate: select max(tid) from Teacher ---------------------------------- Hibernate: insert into Teacher (name, tid) values (?, ?) * */ public void t01ThreeStatus01(){ System.out.println("----------------------------------"); Teacher teacher=new Teacher("孟六爱*");//临时状态(瞬时状态) session.save(teacher);//持久状态 System.out.println("----------------------------------"); tr.commit(); session.evict(teacher);//游离状态 }
四,commit()和flush():
1.区别:
* commit()和flush()的区别
*
* flush():是缓存清理,把缓存中的数据同步到数据库!但是不会持久化
* commit():在执行的时候,会默认执行flush(),之后会持久化
* flush()在执行的时候会进行缓存清理,在缓存清理的时候,会进行脏检查!
2. 什么是脏检查?
* 在一个对象被session管理的时候,会创建这个对象的快照,
* 我们之后commit的时候,会拿当前的对象信息和之前对象的快照进行对比,
* 如果当前对象的属性发生改变,那么现在的对象就是脏对象!
* 脏对象会被同步到数据库中!
3.对一个持久化状态的commit操作:
@Test /* * 写一下commit可以提交属于持久化状态的数据 * Hibernate: select teacher0_.tid as tid1_0_0_, teacher0_.name as name2_0_0_ from Teacher teacher0_ where teacher0_.tid=? Hibernate: update Teacher set name=? where tid=? * */ public void t02CommitAndFlush01(){ Teacher teacher=session.get(Teacher.class,3);/*持久化状态*/ teacher.setName("CommitTest1"); tr.commit(); }
它会在没有update()的情况下也会做修改,并且持久化到数据库
4.对一个持久化状态的flush操作:
@Test /* * 写一下flush可以同步到数据库,但是不会持久化 * Hibernate: select teacher0_.tid as tid1_0_0_, teacher0_.name as name2_0_0_ from Teacher teacher0_ where teacher0_.tid=? * * */ public void t02CommitAndFlush02(){ Teacher teacher=session.get(Teacher.class,3);/*持久化状态*/ teacher.setName("flushTest1"); session.flush(); }
它可以同步到数据库,之后执行完毕之后就回滚掉了,不会持久化,我下面一个例子做证明
5.证明flush只是做同步,不是持久化:
@Test /* * 测试flush是否真实同步到数据库, * 排除缓存,看看是否确实持久化到数据库中还是只是同步,一会就回滚 * Hibernate: select teacher0_.tid as tid1_0_0_, teacher0_.name as name2_0_0_ from Teacher teacher0_ where teacher0_.tid=? ============================CommitTest1 Hibernate: update Teacher set name=? where tid=? Hibernate: select teacher0_.tid as tid1_0_0_, teacher0_.name as name2_0_0_ from Teacher teacher0_ where teacher0_.tid=? ============================flushTest2 * */ public void t02CommitAndFlush03(){ Teacher teacher=session.get(Teacher.class,3);/*持久化状态*/ System.out.println("============================"+teacher.getName());/*CommitTest1*/ teacher.setName("flushTest2"); session.flush(); //清理缓存 session.clear(); //清空缓存 teacher=session.get(Teacher.class,3);/*持久化状态*/ System.out.println("============================"+teacher.getName());/*flushTest2*/ }
五,saveOrUpdate()和merge():
1.区别:
最核心的区别:::::::merge()不会改变对象的状态!!!
* save(): 把瞬时态转换成持久态
* update(): 把游离态转换成持久态
* saveOrUpdate():
* 会根据对象是否有OID来判断执行save还是update
* 如果有oid 执行update
* 如果没有oid 执行save
* merge(): 产生的sql语句和saveOrUpdate有点类似,
* 但是!!!!!
* 01.merge不会改变对象的状态
* 02.当我们的对象处于瞬时状态时,会将对象复制一份到session的缓存中,
* 然后执行save方法,执行insert
2.持久化状态下的saveOrUpdate():
@Test /* * saveOrUpdate * 它会改变状态 * 它会根据有没有oid标识(此java对象的状态),来选择执行save还是update * * 当是update的时候的案例 * Hibernate: select teacher0_.tid as tid1_0_0_, teacher0_.name as name2_0_0_ from Teacher teacher0_ where teacher0_.tid=? Hibernate: update Teacher set name=? where tid=? * */ public void t03SaveOrUpdate01(){ Teacher teacher=session.get(Teacher.class,4);/*持久化状态*/ teacher.setName("SaveOrUpdate01"); session.saveOrUpdate(teacher); /*他走的是update*/ tr.commit(); }
3.临时(瞬时)状态下的SaveOrUpdate():
@Test /* * saveOrUpdate * 它会改变状态 * 它会根据有没有oid标识(此java对象的状态),来选择执行save还是update * * 当是save的时候的案例 * Hibernate: select max(tid) from Teacher Hibernate: insert into Teacher (name, tid) values (?, ?) * * */ public void t03SaveOrUpdate02(){ Teacher teacher=new Teacher("王老师66");//临时状态(瞬时状态) teacher.setName("SaveOrUpdate02"); session.saveOrUpdate(teacher); /*他走的是update*/ tr.commit(); }
4.(临时)瞬时状态下的merge():
@Test /* * Merge和SaveOrUpdate用法类似,但是他不能改变状态,所以有些时候他会有点问题 * 案例一,save * Hibernate: select max(tid) from Teacher Hibernate: insert into Teacher (name, tid) values (?, ?) * */ public void t04Merge01(){ Teacher teacher=new Teacher("Merge01goodnice");//临时状态(瞬时状态) teacher.setName("Merge01"); session.merge(teacher); /*他走的是update*/ tr.commit(); }
5.持久化状态下的merger():
@Test /* * Merge和SaveOrUpdate用法类似,但是他不能改变状态,所以有些时候他会有点问题 * 案例二,update * Hibernate: select teacher0_.tid as tid1_0_0_, teacher0_.name as name2_0_0_ from Teacher teacher0_ where teacher0_.tid=? Hibernate: update Teacher set name=? where tid=? * */ public void t04Merge02(){ Teacher teacher=session.get(Teacher.class,4);/*持久化状态*/ teacher.setName("Merge02"); session.merge(teacher); /*他走的是update*/ tr.commit(); }
6.测试merge()是否会改变对象状态(一):
先准备一个临时对象,做merge操作,之后做update()操作,因为update得操作持久化对象,所以,只需要看报不报错就知道merge会不会改变状态
@Test /* * Merge和SaveOrUpdate用法类似,但是他不能改变状态,所以有些时候他会有点问题 * 案例三,他不会改变状态的案例 它会抛异常,因为不是持久化状态的无法用update,虽然他已经进行了插入操作 Hibernate: select max(tid) from Teacher Hibernate: insert into Teacher (name, tid) values (?, ?) org.hibernate.TransientObjectException: The given object has a null identifier: cn.dawn.day02.entity.Teacher ........ * */ public void t04Merge03(){ Teacher teacher=new Teacher("Merge03");//临时状态(瞬时状态) session.merge(teacher); /*他走的是save,但是没有改变为持久化状态*/ tr.commit(); session.update(teacher); tr.commit(); }
报错,所以merge()没有改变对象状态
7.测试merge()是否会改变对象状态(二):
准备一个临时状态的对象,执行俩次merge(),看是发俩条insert还是有update
@Test /* * Merge和SaveOrUpdate用法类似,但是他不能改变状态,所以有些时候他会有点问题 * 案例四,创建一个对象后,merge一次,之后改变他的属性,再marge一次。。。 他会执行俩次save,因为此对象是临时状态,merge不会改变java对象的状态 Hibernate: select max(tid) from Teacher Hibernate: insert into Teacher (name, tid) values (?, ?) Hibernate: insert into Teacher (name, tid) values (?, ?) ........ * */ public void t04Merge04(){ Teacher teacher=new Teacher("Merge04");//临时状态(瞬时状态) session.merge(teacher); /*他走的是save,但是没有改变为持久化状态*/ teacher.setName("Merge04NewValue"); session.merge(teacher); tr.commit(); }
结论:两条insert,所以merge()不会改变对象状态
作者:晨曦Dawn
转载请注明出处:博客地址:https://www.cnblogs.com/DawnCHENXI/p/9101984.html
如果有错误,请指出!感激不尽!!!!!!!!!!!!!!!!!!!!!!!!!!!!