hibernate框架中对象的状态

hibernate框架中对象的状态

increment:hibernate先会查询数据最大id值,然后在最大值加1在进行保存操作。
hibernate:select max(id) from t_user;
insert into User(hiredate,id)values(?,?)

native 本地数据库的主键策略。

hibernate对象状态
问题:
问题1: 主键生成策略不同,save操作时发送INSERT语句的时机不同.
* native: 在保存对象时,发送SQL.
* increment:在提交事务时,发送SQL.
问题2: 删除对象的时候,没有立刻发生DELETE语句,而是在提交事务的时候发送的.
问题3: 为什么在事务环境下,通过get方法得到的对象,只要修改了属性值,会发生UPDATE语句.

hibernate执行流程,不能从发送sql角度去理解,应该从对象状态方向去理解。

session方法改变对象什么状态?

1.对象状态
临时状态/瞬时态(transient):
刚刚用new语句创建,没有被持久化,不处于session中。
特点:没有oid,不在session当中

持久化状态(persistent):
已经被持久化,加入到session的缓存中。
特点:有oid,在session当中

脱管态/游离状态(detached):
已经被持久化,但不处于session中。
特点:有oid,不在session当中

删除状态(removed):
对象有关联的ID,并且在Session管理下,但是已经计划被删除。
特点:有oid,在session当中,最终的效果是被删除.
可以不考虑,没有什么意义。

判断规则:
1): 对象是否有OID;———可以理解对象的Id,数据库中的主键id
2): 判断对象是否被Session所管理(在一级缓存中).

2.临时/瞬时状态
没有oid,没有被session管理
1):new语句刚创建了一个对象.
Session session = HibernateUtils.getSession();
User user = new User();
user.setName(“tom3”);
user.setSalary(BigDecimal.valueOf(20000.0));
user.setHiredate(new Date());
//———————–此时的user对象是临时瞬时状态:没有id,不被session管理
//——————主键策略为 native,保存的时候就发送sql语句
//开启事物
session.getTransaction().begin();
System.out.println(“—————–“);
session.save(user);
System.out.println(“——————–“);
//提交事物
session.getTransaction().commit();
session.close();

控制台打印:
	-----------------
	Hibernate: 
		insert 
		into
			t_user
			(uname, usalary, uhiredate) 
		values
			(?, ?, ?)
	--------------------
2)情况2):删除状态的对象,在事务提交之后,对象处于临时状态.
  临时状态是没有ID的,测试可以打印该对象的ID,发现存在ID.
   --->设置hibernate.cfg.xml的属性:use_identifier_rollback=true

3.删除状态
特点:此时有OID,被Session所管理中,
但是最终会被删除(我们不关心删除状态的对象).
删除状态的对象必须等到session刷新(flush),
事务提交时才真正从数据库中删除
1)游离状态到删除状态
游离状态:有oid,不被session管理
//游离状态—>删除状态
User u = new User();
u.setId(1L);
//————–此时对象状态为游离托管状态:有id,没被session管理
Session session = HibernateUtils.getSession();
session.getTransaction().begin();

	session.delete(u);// 对象状态为删除状态-----删除的时候在事物提交的时候才发送sql
	System.out.println("-----------------");
	session.getTransaction().commit();
	System.out.println("-------------------");

	控制台打印:
		-----------------
		Hibernate: 
			delete 
			from
				t_user 
			where
				uid=?
		-------------------

2)持久状态编程删除状态

	Session session = HibernateUtils.getSession();
	User user = (User) session.get(User.class, 6L);//持久状态,有id、被session管理
	
	session.getTransaction().begin();
	
	session.delete(user);
	System.out.println("-------------------");
	session.getTransaction().commit();
	System.out.println("-------------------");

	控制台打印:
		Hibernate: 
			select
				user0_.uid as uid1_0_0_,
				user0_.uname as uname2_0_0_,
				user0_.usalary as usalary3_0_0_,
				user0_.uhiredate as uhiredat4_0_0_ 
			from
				t_user user0_ 
			where
				user0_.uid=?
		-------------------
		Hibernate: 
			delete 
			from
				t_user 
			where
				uid=?
		-------------------

4.持久化状态):特点:有OID,被Session所管理(在一级缓存中).
情况1)调用save方法把临时状态变为持久状态
情况2)调用save方法把游离托管状态状态变为持久对象
保存一个对象之后,提交事务/关闭Session,此时对象处于游离状态,
再创建新的Session来保存该对象.
情况3):get和load方法返回的是持久化对象.
情况4):Query.list方法返回的是持久化对象,在处理大数据量的时候,
需要及时清理一级缓存(分页查询).
情况5):update方法把游离对象变成持久化对象.

5.Detached(游离状态/托管状态):特点:有OID,但是不被Session所管理(不在一级缓存中).
情况1):session.close()方法把所有的持久化对象变成游离对象.
情况2):session.clear()方法把所有持久化对象变成游离对象.
情况3):session.evivt(Object)方法把制定的持久化对象变成游离对象.
情况4):使用new创建对象,并设置OID(数据库存在该ID):临时状态–>游离状态;

总结:
new出来的对象—–>临时状态—save()/saveorupdate()/persist()—->持久状态

  ------get()/query/iteratro/load-----> 持久对象----close()/clear()/evict()--------------> 游离状态
		                                      <--save()/saveorupdate()/update()/merge()----
   
   
   
   持久状态-----delete()---->删除状态
							 /
							/
   游离状态-----delete()----

1:save方法只需要把对象从临时变成持久化状态,只需要找到OID即可.不同的ID生成策略,

2: 因为delete方法仅仅只是把游离对象或持久化对象变成删除状态,并不负责发生SQL.
session中的方法仅仅是改变对象的状态,不发SQL:

3: 持久化对象的属性真正发生改变时,才会发生UPDAE语句.


发生SQL的时机:
默认情况下,在事务提交时,会自动去数据库同步这一次对象变化对应的SQL.
事务提交的时候,到底发送了什么SQL?
1:临时状态–>持久化状态:发送INSERT.
2:持久化状态/游离状态—>删除状态:发送DELETE.
3:游离状态–>持久化状态:发送SQL(可能是INSERT或UPDATE).
4:脏的持久化对象同步到数据库.(session快照)
5:session.flush()方法可以手动同步数据库.

总结:由session的持久化方法修改对象的状态,
在同步session数据的时候(默认是提交事务,也可以是flush),
session再同步脏数据(一级缓存和一级快照中数据是否一致,不一致发送sql,)到数据库,完成内存对象和数据库的同步.

 session方法负责改变状态,事务负责同步数据.
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/100799.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • ubuntu服务器硬件配置

    ubuntu服务器硬件配置1.系统cat/etc/issue2.内存free-h3.硬盘df-h4.CPUcat/proc/cpuinfocpu个数cat/proc/cpuinfo|grep”physicalid”|sort|uniq|wc-l

    2022年10月21日
  • linux系统对硬盘分区_centos怎么分区

    linux系统对硬盘分区_centos怎么分区Linux操作系统磁盘分区操作及原理,Linux系统网络环境的配置方法。

  • 提示用微信浏览器打开_浏览器如何设置消息提醒

    提示用微信浏览器打开_浏览器如何设置消息提醒概述大体思路是这样的:先建一个蒙版,蒙版放上提示的信息,然后写一个js,在页面打开的时候判断是不是用微信自带的浏览器打开的,如果是就打开蒙版,提示用浏览器打开基本页面添加蒙版添加js最终效果

  • epp和edr_一文读懂分频器

    epp和edr_一文读懂分频器一文读懂什么是EPP、EDR、CWPP、HIDS及业内主流产品当前终端安全概念包括:针对云工作负载保护平台cwpp、端点防护平台epp和终端全检测响应平台edr。HIDS品类(长亭牧云、青藤万相)更倾向于CWPP的落地产品。1、EPP与EDR如果通俗的讲可以理解成传统防病毒和下一代防病毒软件(其实EPP之前的阶段才是传统杀毒),但实际上EDR是个方案,深信服EDR、奇安信天擎、安恒明御对外宣传都是EDR产品,从产品能力严谨来讲,这些产品属于EPP+EDR的方案结合产物,其中EPP解决…

    2022年10月22日
  • AutoFac使用

    AutoFac使用AutoFac使用方法设置读取配置文件的方法AutoFacConfig.cs:需要安装引用Autofac3.5.2Autofac.Configuration3.3.0=&gt;ConfigurationSettingsReaderAutofac.Owin4.0.0Autofac.WebApi24.1.0Autofac.WebApi2.Owin4.0.0代码publicsta…

    2022年10月27日
  • c# restsharp官网_hbase shell put

    c# restsharp官网_hbase shell putusingSystem;usingSystem.Net.Http;usingSystem.Threading.Tasks;usingSystem.Collections.Generic;usingNewtonsoft.Json;usingSystem.Net;usingSystem.IO;usingSystem.Text;usingRestSharp;namespaceHttpClientQuery{classPageInfo{publicintstar

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号