Hibernate学习(4)- Hibernate对象的生命周期
1、Hibernate对象的生命周期(瞬时状态、持久化状态、游离状态)
1、瞬时状态(Transient):
使用new操作符初始化的对象就是瞬时状态,没有跟任何数据库数据相关联;
2、持久化状态(Parsistent):
如果对象与Session对象关联起来,且该对象对应到数据库记录,则称该对象处于持久化状态。
3、游离状态(Detached)
Session被关闭或调用了Session的evict或clear方法把它从Session中移除了,则该对象脱离了Session的管理,持久化状态变成游离状态,这表示该对象不在和数据库保持同步,不受hibernate管理。
2、三种状态的比较
Transient:瞬时状态的对象只存在于内存中。
Parsistent:持久状态的对象分别存在于内存、session对象、数据库之中。
Detached:游离状态的对象存在于内存、数据库之中,但不在session对象中。
3、三种状态之间的转换

1、转化路径详解
1)瞬时对象(Transient)
new 创建 (无->Transient) ,即:对象通过构造方法成为瞬时态;
2)持久对象(Persistent)
1.1)get/load (无->Persistent),即:对象可以由session的load或get方法直接成为持久态;
get通过类名和id从数据库读取指定记录,无匹配记录返回null。
load通过类名和id从数据库读取指定记录,无匹配记录抛ObjectNotException异常。
1.2)save/saveOrUpdate/persist (Transient->Persistent) ,即:瞬时态对象可以通过save,saveOrUpdate或persist方法成为持久态;
1.3)update/saveOrUpdate(Detached->Persistent),即:游离态对象则可以通过update,saveOrUpdate成为持久态;
3)游离对象(Detached)
游离态只能由持久态转换而来(Persistent->Detached),通过close、clear、evict方法实现。
evict--把某个对象从session中移除,变为游离态;
clear--把所有对象从session中移除,对象全部变为游离态;
close--关闭session,其中的对象全部变为游离态;
2、几种方法的详解
1、get 与 load
都是从数据库中加载数据封装为java对象,直接变为持久态;
2、save,update与saveOrUpdate
save是将瞬时态转为持久态;
update是将游离态转为持久态;
saveOrUpdate可以说是两者的综合,它执行时先判断对象的状态(主要是通过有无主键判断的),若是自由态,则save,若是游离态,则update;
3、save与persist
两者都是将对象由瞬时态转为持久态,但返回值不同:save返回主键值,而persist不返回;
4、实例,详见注解状态(封装的工具类详见Demo)
package com.demo.test; import org.hibernate.Session;
import org.junit.Test; import com.demo.Utils.HibernateUtil;
import com.demo.pojo.User; public class SessionTest { @Test
public void saveTest() {
Session session = null;
User user = null;
try {
session = HibernateUtil.currentSession();
session.beginTransaction();
//创建瞬时状态对象
user = new User();
user.setName("东方不败");
user.setPwd("123456");
//user仍是一个瞬态对象 //持久状态,user被session管理,并且id有值 -- oid
//此时user已变成 持久态对象
session.save(user); //在持久状态下脏数据检查:当提交事务,清理缓存时发现session中的数据与数据库的数据不一致时,会把session的数据更新到数据库
//保存以后,再修改对象,会产生多条sql语句,效率降低,所以在save前修改数据
user.setName("西方求败");
session.getTransaction().commit();
}catch (Exception e) {
e.printStackTrace();
}finally {
HibernateUtil.closeSession();
}
//close、clear、evict方法都会使持久态对象 变成 游离状态 -- user
user.setName("令狐冲"); try {
session = HibernateUtil.currentSession();
session.beginTransaction();
//update使对象变成 持久状态
session.update(user);
session.getTransaction().commit();
}catch (Exception e) {
e.printStackTrace();
}finally {
HibernateUtil.closeSession();
}
} @Test
public void getTest() {
Session session = null;
try {
session = HibernateUtil.currentSession();
//get后变为持久状态
//get方法会立即查询该对象:范围(查询不到去下一级查询)从session -> sessionFactory ->数据库
//get方法找不到对象,不会有异常,返回nul
User user = session.get(User.class, 7L);
System.out.println(user.toString()); User user2 = session.get(User.class, 17L);
System.out.println(user2);
}catch (Exception e) {
e.printStackTrace();
}finally {
HibernateUtil.closeSession();
}
} @Test
public void loadTest() {
Session session = null;
try {
session = HibernateUtil.currentSession();
//load后变为持久状态(这里加载机制与get不同,稍后我会单独写一篇文章 介绍 get与load 的区别)
//对象不存在时,抛出异常
User user = session.load(User.class, 7L);
System.out.println(user.toString()); User user2 = session.load(User.class, 17L);
System.out.println(user2.toString());
}catch (Exception e) {
e.printStackTrace();
}finally {
HibernateUtil.closeSession();
}
} @Test
public void clearTest() {
Session session = null;
User user = null;
try {
session = HibernateUtil.currentSession();
session.beginTransaction();
user = session.get(User.class,7L);
System.out.println(user.getName());
session.getTransaction().commit();
session.clear();
//游离状态,不被session管理,数据库不会被更改
user.setName("任我行");
System.out.println(user.getName());
}catch (Exception e) {
e.printStackTrace();
}finally {
HibernateUtil.closeSession();
}
} @Test
public void updateTest() {
Session session = null;
User user = null;
try {
session = HibernateUtil.currentSession();
session.beginTransaction();
//瞬时状态
//手动构造对象 也可以修改,但是需要指定所有的属性值,否则,不设置的对象会置空,不建议这样update数据
/*user = new User();
user.setId(7L);
user.setName("盈盈");
session.update(user);*/
//一般这样修改
//先去查询
user = session.get(User.class,7L);
if(user != null) {
user.setName("盈盈");
session.update(user);
}
session.getTransaction().commit();
}catch (Exception e) {
e.printStackTrace();
}finally {
HibernateUtil.closeSession();
}
} @Test
public void deleteTest() {
Session session = null;
User user = null;
try {
session = HibernateUtil.currentSession();
session.beginTransaction();
//瞬时状态
//手动构造一个对象,指定主键是可以删除数据的,不建议这样搞
/*user = new User();
user.setId(7L);
session.delete(user);*/
//应该这样删除数据
//从数据先加载该对象,然后删除,可以避免异常
user = session.get(User.class, 7L);
if(user != null) {
session.delete(user);
}
session.getTransaction().commit();
}catch (Exception e) {
e.printStackTrace();
}finally {
HibernateUtil.closeSession();
}
} }
PS:源码地址 https://github.com/JsonShare/hibernate-demo
PS:原文地址 http://www.cnblogs.com/JsonShare/p/8686078.html
Hibernate学习(4)- Hibernate对象的生命周期的更多相关文章
- java之hibernate之session中对象的生命周期
1. session是用来执行对象的crud操作,并且session是对象事务工厂.session是线程级别的,所以生命周期比较短. 2.session中对象的生命周期图: 3.session中对象的 ...
- [原创]java WEB学习笔记94:Hibernate学习之路---session 的管理,Session 对象的生命周期与本地线程绑定
本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...
- hibernate 持久化对象的生命周期 2.1
持久化对象的生命周期 瞬态(自由态) 表示对象在内存中存在,在数据库中没有数据相关,比如刚刚new出来的一个对象 持久态 持久态指的是持久化对象处于由Hibernate管理的状态,这种状态下持久化对象 ...
- hibernate 持久化对象的生命周期
持久化对象的生命周期 瞬态(自由态) 表示对象在内存中存在,在数据库中没有数据相关,比如刚刚new出来的一个对象 持久态 持久态指的是持久化对象处于由Hibernate管理的状态,这种状态下持久化对象 ...
- Hibernate中Java对象的生命周期
一个对象的出生源于我们的一个new操作,当我们使用new语句创建一个对象,这个对象的生命周期就开始了,当我们不在有任何引用变量引用它,这个对象就的生命就此结束,它占用的内存就可以被JVM的垃圾回收器回 ...
- [原创]java WEB学习笔记81:Hibernate学习之路--- 对象关系映射文件(.hbm.xml):hibernate-mapping 节点,class节点,id节点(主键生成策略),property节点,在hibernate 中 java类型 与sql类型之间的对应关系,Java 时间和日期类型的映射,Java 大对象类型 的 映射 (了解),映射组成关系
本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...
- [原创]java WEB学习笔记47:Servlet 监听器简介, ServletContext(Application 对象), HttpSession (Session 对象), HttpServletRequest (request 对象) 监听器,利用listener理解 三个对象的生命周期
本博客为原创:综合 尚硅谷(http://www.atguigu.com)的系统教程(深表感谢)和 网络上的现有资源(博客,文档,图书等),资源的出处我会标明 本博客的目的:①总结自己的学习过程,相当 ...
- Hibernate学习笔记-Hibernate HQL查询
Session是持久层操作的基础,相当于JDBC中的Connection,通过Session会话来保存.更新.查找数据.session是Hibernate运作的中心,对象的生命周期.事务的管理.数据库 ...
- hibernate 学习 五 hibernate核心接口
一 Configuration接口 Configuration对象只存在于系统的初始化阶段.配置相关. 配置文件可以使用默认的路径,也可以指定路径. Configuration config = ne ...
随机推荐
- CSS的继承性与优先级
一.CSS的继承性 在CSS中不可继承的属性:display.margin.padding.border.background.width.min-width.max-width.height.min ...
- 校园网IPv6加速
对于广大学生来说,上网是一件很纠结的事情,校园网要么按时间计费,要么按流量计费,要么是校园宽带.按时间计费速度慢,按流量计费费用高,校园宽带还不能共享,只能电脑开热点给手机上网.有没有既能提高网速又经 ...
- JS对select动态添加options操作(主流浏览器兼容)
之前项目中,遇到一个表单提交的页面,里面有多级下拉框联动的复杂逻辑,因此当时在做的过程中也是学到了不少容易出现问题的地方,下面就整理下当时遇到的一些关于下拉框的操作,并指出其中的一些注意点和坑: 有如 ...
- (luogu4180) [Beijing2010组队]次小生成树Tree
严格次小生成树 首先看看如果不严格我们怎么办. 非严格次小生成树怎么做 由此,我们发现一个结论,求非严格次小生成树,只需要先用kruskal算法求得最小生成树,然后暴力枚举非树边,替换路径最大边即可. ...
- [BZOJ2752][HAOI2012]高速公路
BZOJ Luogu sol 看上去是道数学期望题但实际上是个傻逼数据结构 首先答案的形式应该就是 \[\frac{\mbox{[l,r]区间内的子区间权值之和}}{\mbox{[l,r]区间内的子区 ...
- Eclipse增强代码提示插件Code Recommenders安装,顺便说说Eclipse插件安装方法
1.为什么用Code Recommenders 在用过Intelij Idea后,发现它的自动代码提示非常智能,可以敲关键字就能提示,但是因为公司用的是Eclipse, 所以想找有没有这个插件能增强代 ...
- asp.net core 五 SignalR 负载均衡
SignalR : Web中的实时功能实现,所谓实时功能,就是所连接的客户端变的可用时,服务端能实时的推送内容到客户端,而不是被动的等待客户端的请求.Asp.net SignalR 源码 ...
- Ceph常见问题百科全书
Ceph是目前炙手可热的一个统一分布式存储系统,具有优异的性能.可靠性.可扩展性.其可轻松扩展到数 PB 容量, 支持多种工作负载的高性能(每秒输入/输出操作[IOPS]和带宽),具有极其高的可靠性. ...
- python 检测nginx状态,若无法访问发邮件通知
应用场景:用来检测网站可用性,访问失败,则发邮件通知 #!/usr/bin/env python import urllib2,time,smtplib,string,logging from con ...
- PAT乙级-1057. 数零壹(20)
给定一串长度不超过105的字符串,本题要求你将其中所有英文字母的序号(字母a-z对应序号1-26,不分大小写)相加,得到整数N,然后再分析一下N的二进制表示中有多少0.多少1.例如给定字符串" ...