Hibernate-ORM:06.Hibernate中三种状态
------------吾亦无他,唯手熟尔,谦卑若愚,好学若饥-------------
本篇博客主要叙述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
如果有错误,请指出!感激不尽!!!!!!!!!!!!!!!!!!!!!!!!!!!!
Hibernate-ORM:06.Hibernate中三种状态的更多相关文章
- Hibernate的工作流程以及三种状态(面试题)
Hibernate的工作流程以及三种状态 部分转载自:http://www.cnblogs.com/fifiyong/p/6390699.html Hibernate的工作流程: 1. 读取并解析配置 ...
- Hibernate(六)__对象的三种状态
瞬时(transient):数据库中没有数据与之对应,超过作用域会被JVM垃圾回收器回收,一般是new出来且与session没有关联的对象. 持久(persistent):数据库中有数据与之对应,当前 ...
- hibernate(二)一级缓存和三种状态解析
序言 前一篇文章知道了什么是hibernate,并且创建了第一个hibernate工程,今天就来先谈谈hibernate的一级缓存和它的三种状态,先要对着两个有一个深刻的了解,才能对后面我要讲解的一对 ...
- Hibernate学习笔记2.5(Hibernate核心开发接口和三种状态)
1.configuration(配置信息管理,产生sessionfactory) sessionfactory管理一系列的连接池 opensession 永远打开新的,需要手动close getcur ...
- Hibernate学习2--对象的三种状态以及映射关系的简单配置
上篇hibernate的博客总体简单梳理了对象持久化的一些思想以及hibernate中对象持久化化的方法,下面说说对象持久化过程的三种状态. 一.hibernate缓存的概念 1.session与缓存 ...
- Hibernate的工作流程以及三种状态
Hibernate的工作流程: 1. 读取并解析配置文件 2.读取并解析映射信息,创建SessionFactory 3. 打开Sesssion 4.创建事务Transation 5. 持久化操作 6. ...
- Hibernate之对象的三种状态
Hibernate之Java对象的三种状态 一.简述 本博文大部分的思想和内容引子CSND上名为 FG2006 这位大神的文章,连接地址为:http://blog.csdn.net/fg2006/ar ...
- Hibernate 系列 07 - Hibernate中Java对象的三种状态
引导目录: Hibernate 系列教程 目录 1. Java对象的三种状态 当应用通过调用Hibernate API与框架发生交互时,需要从持久化的角度关注应用对象的生命周期. 持久化声明周期是Hi ...
- Hibernate中Java对象的三种状态
Hibernate中Java对象的三种 ...
随机推荐
- 在Hibernate单向一对多关联关系中的org.hibernate.StaleStateException 异常。
具体异常如下: Caused by: org.hibernate.StaleStateException: Batch update returned unexpected row count fro ...
- PIL 一秒切九图 朋友圈发图神器
注意图片像素返回值是(宽度,高度),pil填像素点坐标原点左上角. 判断像素点是否在圆方程中. import numpy as np from PIL import Image file = inpu ...
- LA 4731 蜂窝网络
题目链接:https://vjudge.net/problem/UVALive-4731 题意: n 个 数,分成 w 组,求整个区间的数学期望的最小值: 一个区间的数学期望公式给出:一个区间的和 * ...
- 【[USACO15FEB]审查(黄金)Censoring (Gold)】
从原来的单串匹配变成了多串匹配 好像也没什么特别不一样的地方 原来的做法是搞一个栈,之后一旦匹配到就往前弹栈 做法也一样 但是在\(AC\)自动机上暴力跳\(fail\)是要\(T\)的 我们并没有必 ...
- HDU 2157 How many ways?? 【矩阵经典8】
任意门:http://acm.hdu.edu.cn/showproblem.php?pid=2157 How many ways?? Time Limit: 2000/1000 MS (Java/Ot ...
- 2018.12.17 struts.xml 配置自定义拦截器配置
自定义拦截器有三个步骤哦 <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE struts PU ...
- E: Unable to locate package
E: Unable to locate package apt-get不能定位到包,有两种情况,一种是自己输入的包名字错误,确实找不到.另一种可能是执行sudo apt-get install之前更换 ...
- Visual Studio C++ Win32控制台应用程序,Win32项目,MFC的区别
背景 Visual Studio C++ 创建新项目蹦出来如下选项: Win32控制台应用程序,Win32项目,MFC有什么区别? 正文: Win32控制台,没有界面,命令行执行生成的文件则直接在后台 ...
- Ajax去除缓存
1.在Ajax发送请求钱加上anyAjaxObj.setRequestHeader("If-Modeified-Since","0").2.在Ajax发送请求钱 ...
- mycat的安装及配置文件应用
table:逻辑一 mycat的安装 1 基于jdk运行 2 获取安装包 3 解压 tar -xf Mycat***.tar.gz 4 测试运行 mycat的根目录中bin保存了mycat的核心命令文 ...