三.通过hibernate操作对象
要理解hibernate是如何操纵对象的首先要知道hibernate的缓存机制,缓存的作用是什么我就这里我就不说了,我们说说Session进行脏检查和清理缓存的机制,这一点非常重要:
Session是如何进行脏检查的呢,当一个Customer对象呗加入到Session缓存中时,Session会为Customer对象的值类型的属性复制一份快照,Session清理缓存的时候,会先进行脏检查,比较Customer的当前属性和他的快照,看是否发生了变化,如果有变化就称这个对象时脏对象,那么Session就会根据最新的属性来执行相关的sql语句,从而同步数据库。
上面的值类型是指类似name,age等等java基本类型的类型。另一种就是实体类型。当然,Session也不是在你以改变值的时候就立即执行sql语句,他会在清理缓存的时候进行执行,这样也使得Session能将多个修改合并成一个sql语句,一起提交到数据库从而提高性能。
有个例外情况是,如果某对象使用native生成器来生成OID,那么当调用Session的save()方法时,积极不会等待清理缓存的时候才执行这个语句了,而是在save()的时候就立即执行insert语句。
1.Session默认清理缓存的时间点:
A.当应用程序调用事务的commit()方法的时候,commit()会先清理缓存,然后向数据库提交事务。
B.当应用程序执行一些查询操作时,如果缓存中持久化对象的属性已经发生变化,就会先清理缓存,使得Session缓存和数据库已进行了同步,从而保证查询结果返回的是正确的数据。
C.当程序显示调用Session的flush()方法的时候。Session一般不会显式调用flush()方法,一般是在某个插入,删除或更新操作会引发数据库的某个触发器的时候才显式调用flush()方法。
注意:Session的flush()方法和commit()方法的区别:
flush()方法进行清理的时候不会提交事务,也就是不会将更新的属性同步到数据库,而commit()会先调用flush()方法清理缓存,然后再提交事务。
2.hibernate中对象的状态:
关于对象在hibernate中的状态有的说是3种,有的说是4种,我坚持是4中,分别是临时状态,持久状态,游离状态,删除状态,下面是转换图:
<!--[endif]-->
对象在hibernate持久化层的状态转换图
3.Session接口的用法:
3.1当对象处于持久化状态时,不允许任意修改它的OID,否则会抛出hibernateException异常。在这里我们也建议在定义持久化类时,把它的setID()方法设置为private类型。禁止外部程序访问该方法。
3.2 save()方法是用来持久化一个临时对象的,在程序中把一个持久化对象传给save()方法是多余的。
3.3 persist()方法和save方法都能够把一个临时对象转换成持久化对但是他们的区别在于:persist()方法不保证会立即为持久化对象的OID赋值,而是有可能在Session清理缓存时才为IOD赋值。而且在事务以外调用persist()方法将不会计划执行insert语句,而save()方法不管在事务以内还是以外都会计划执行SQL insert语句.
3.4 Session的load()方法和get()方法的相同和区别:
get()和load()方法都能根据给定的OID从数据库中加载一个持久化对象,这个两个方法的一个区别在于,当数据库不存在与OID对应的记录时,load()方法抛出ObjectNotFoundException异常,而get()方法返回null.
另一个更重要的区别是,他们的检索策略不同,load会采用延迟检索策略记载持久化对象,除非把<class>元素的lazy=””属性设置为true,load()方法才会采用立即检索策略,而get()方法会忽略lazy的值,直接使用立即检索策略。这两种方法适合不同的场合,比如
如果要加载一的对象的目的是为了访问它的各个属性,那么采用get()方法,如果加载一个对象的目的是为了删除它,或者为了建立与别的对象的关联关系,可以用load()方法,如下:
tx = session.beginTransaction();
Customer customer=(Customer)session.load(Customer.class,new Long(1));
Order order=(Order)session.load(Order.class,new Long(1));
order.setCustomer(customer);//建立custermer到order的关联
tx.commit();
3.5 Session的saveOrUpdate()方法,如果传入的参数是临时对象,就调用save()方法,如果是游离对象就调用update()方法。
3.6 Session的merge()方法,merge()方法的产生源于update()方法的不给力,在实行update()方法时,如果在Session缓存中依旧存在相同OID的持久化对象或者在数据库不存在相应的记录,update()会抛出异常。
而merge()不会,对于前者merge()会将该游离对象复制到该缓存中的持久化对象中,然后计划执行update语句,并返回持久化对象的引用;
对于后者,如果数据库不存在该记录,merge()方法会创建一个新的该对象,并把这个游离对象复制到该对象中,然后调用save()方法持久化这个对象,返回这个对象的引用。
3.7 一般把<many-to-one>元素的casecade属性设置为”none”而不是”save-update”,是为了防止保存一个子类时,子类会级联到上级,将本处于游离状态的上级做一个没有任何改变的update sql语句。(因为update()会将游离状态的对象转换为持久状态的对象),以此提高性能。
3.7 Session与触发器协同工作
向数据库进行保存,更新,或删除对象时,如果技法数据库中的某个触发器,常常会带来一个问题,那就是Session缓存中的吃就算对象无法与数据库中的数据保持同步。比如CUSTOMERS表有个REGISTERED_TIME字段,如果定义该字段为数据库自动将当前时间作为其值,当hibernate保存一个对象的时候就会激发这个触发器。当然,既然这个字段在数据库就自动生成了,我们也不需要给这个字段进行更新和插入了,所以我们将这个字段对于的<property> 元素的insert属性和update属性都设置为false.而现在的问题是,如果插入一个对象后要取出这个对象的时间返回的却一直是null,解决办法是
save(customer)
session.flush();
session.refresh();
在save()后加上这里这两个方法后,flush会执行save()的insert语句,然后又立即调用refresh(),方法,重新从数据库加载刚刚保存的Customer对象。
另外如果数据库定义了update()和saveOrUpdate()方法,则要谨慎使用了,因为当处理游离对象时,hibernate始终会执行update语句使之成为持久状态的对象。而这一操作可能会导致激发update触发器,而因为属性并没有变化,所以这一触发器是没有意义的。为避免这种情况,应该在对于<class>元素中添加
select-before-update=”true”
该属性可以使之在update 之前调用select进行比较再决定是否更新到数据库。
4. 批量处理数据
主要有以下方式
A.通过session来进行批量操作
B.通过StatelessSession
C.通过HQL来进行批量操作
D.直接通过JDBC来操作
注意:进行批量操作时,建议关闭第二季缓存,否则会影响系统性能
4.1 通过Session来进行批量操作
它的做法就是在处理完一个对象或者小批量对象后,立刻调用flush()方法清理缓存。
通过session来处理会受到以下约束:
A.需要再hibernate的配置文件中设置JDBC单次批量处理的数目,合理的取值为10-50个,如下配置
hibernate.jdbc.batch_size=20
B.如果对象采用identity标识符生成器,则hibernate无法在JDBC层进行批量处理。
C.为提高系统性能,建议在批量操作时关闭二级缓存(默认关闭),否则在第一季缓存中创建的对象还要复制到二级缓存中然后再保存到数据库。
4.1.1 批量插入代码:
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
for (int i = 0; i < 100000; i++) {
Customer customer=new Customer(.....);
session.save(customer);
if(i % 20 == 0){
session.flush();//清理缓存,执行sql insert语句
session.clear();//清空缓存中的Customer对象
}
}
4.1.1 批量更新代码:
更新我们使用ScrollableResults对象
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
ScrollableResults customers=session.createQuery("form Customer").scroll(ScrollMode.FORWARD_ONLY);
int count=0;
while(customer.next()){
Customer customer = customers.get(0);
customer.setAge(customer.getAge()+1);
if (++count %20 ==0) {
session.flush(); //清理缓存,执行sql update语句
session.clear(); //清空缓存中的Customer对象
}
}
解析:上面scroll返回的ScrollableResults对象其实并不包含任何Customer对象,仅包含用于在线定位数据库中的CUSTOMERS记录的邮编,当程序遍历访问ScrollableResults中的特定元素时才会到数据库加载相应的Customer对象。
4.2 通过StatelessSession来进行批量操作
进行批量操作时将大量对象放到session中浪费大量内存空间,作为一种替代方案可以采用无状态的StatelessSession来进行批量操作。
Session session = sessionFactory.openSession();
Transaction tx = session.beginTransaction();
ScrollableResults customers=session.getNamedQuery("GetCustomers").scroll(ScrollMode.FORWARD_ONLY);
int count=0;
while(customer.next()){
Customer customer = customers.get(0);
customer.setAge(customer.getAge()+1);
session.update(customer);
}
<!--[endif]-->
4.3 另外两个,HQL和JDBC的方法就不说了,很简单,就是通用的hql和sql语句来做。
相关推荐
1. **O/R Mapping**:对象关系映射是Hibernate的核心,它将Java对象与数据库中的记录对应起来,使得开发者可以通过操纵对象来间接操作数据库。O/R Mapping的主要目的是消除对象世界与关系数据库世界的差异,提高开发...
ORM框架如Hibernate,将Java对象与数据库中的表格进行映射,这样开发者就可以通过操纵对象而不是SQL语句来完成数据的存取,极大地提高了开发效率和代码的可维护性。 这份"hibernate学习笔记"涵盖了以下关键知识点:...
##### 知识点1:创建和操纵对象 - **创建对象**:`new 类型(参数列表)`。此语句创建了一个指定类型的对象实例。 - 示例:`String s = new String("Hello World");` - 其中,`new`是关键字,用于创建新的对象实例...
Hibernate是一种开源的对象关系映射(ORM)框架,它主要用于简化Java应用程序与关系数据库之间的交互。ORM框架的主要目标是消除直接的JDBC操作,通过提供一套高级API,使得开发者可以使用面向对象的方式来操作数据库...
uniapp实战商城类app和小程序源码,包含后端API源码和交互完整源码。
本课程是 PHP 进阶系列之 Swoole 入门精讲,系统讲解 Swoole 在 PHP 高性能开发中的应用,涵盖 协程、异步编程、WebSocket、TCP/UDP 通信、任务投递、定时器等核心功能。通过理论解析和实战案例相结合,帮助开发者掌握 Swoole 的基本使用方法及其在高并发场景下的应用。 适用人群: 适合 有一定 PHP 基础的开发者、希望提升后端性能优化能力的工程师,以及 对高并发、异步编程感兴趣的学习者。 能学到什么: 掌握 Swoole 基础——理解 Swoole 的核心概念,如协程、异步编程、事件驱动等。 高并发处理——学习如何使用 Swoole 构建高并发的 Web 服务器、TCP/UDP 服务器。 实战项目经验——通过案例实践,掌握 Swoole 在 WebSocket、消息队列、微服务等场景的应用。 阅读建议: 建议先掌握 PHP 基础,了解 HTTP 服务器和并发处理相关概念。学习过程中,结合 官方文档和实际项目 进行实践,加深理解,逐步提升 Swoole 开发能力。
matlab齿轮-轴-轴承系统含间隙非线性动力学 基于matlab的齿轮-轴-轴承系统的含间隙非线性动力学模型,根据牛顿第二定律,建立齿轮系统啮合的非线性动力学方程,同时也主要应用修正Capone模型的滑动轴承无量纲化雷诺方程,利用这些方程推到公式建模;用MATLAB求解画出位移-速度图像,从而得到系统在不同转速下的混沌特性,分析齿轮-滑动轴承系统的动态特性 程序已调通,可直接运行 ,关键词:Matlab;齿轮-轴-轴承系统;含间隙非线性动力学;牛顿第二定律;动力学方程;修正Capone模型;无量纲化雷诺方程;位移-速度图像;混沌特性;动态特性。,基于Matlab的齿轮-轴-轴承系统非线性动力学建模与混沌特性分析
2024年移动应用隐私安全观测报告.pdf
本电影评论网站管理员和用户。管理员功能有个人中心,用户管理,电影类别管理,电影信息管理,留言板管理,论坛交流,系统管理等。用户可以对电影进行评论。因而具有一定的实用性。本站是一个B/S模式系统,采用SSM框架,MYSQL数据库设计开发,充分保证系统的稳定性。系统具有界面清晰、操作简单,功能齐全的特点,使得电影评论网站管理工作系统化、规范化。 本系统的使用使管理人员从繁重的工作中解脱出来,实现无纸化办公,能够有效的提高电影评论网站管理效率。 关键词:电影评论网站;SSM框架;MYSQL数据库 1系统概述 1 1.1 研究背景 1 1.2研究目的 1 1.3系统设计思想 1 2相关技术 2 2.1 MYSQL数据库 2 2.2 B/S结构 3 2.3 Spring Boot框架简介 4 3系统分析 4 3.1可行性分析 4 3.1.1技术可行性 4 3.1.2经济可行性 5 3.1.3操作可行性 5 3.2系统性能分析 5 3.2.1 系统安全性 5 3.2.2 数据完整性 6 3.3系统界面分析 6 3.4系统流程和逻辑 7 4系统概要设计 8 4.1概述 8 4.2系统结构 9 4.
2023-04-06-项目笔记-第四百三十六阶段-课前小分享_小分享1.坚持提交gitee 小分享2.作业中提交代码 小分享3.写代码注意代码风格 4.3.1变量的使用 4.4变量的作用域与生命周期 4.4.1局部变量的作用域 4.4.2全局变量的作用域 4.4.2.1全局变量的作用域_1 4.4.2.434局变量的作用域_434- 2025-03-13
基于STM32的流量计智能流速流量监测、水泵报警系统(泵启动 1100027-基于STM32的流量计智能流速流量监测、水泵报警系统(泵启动、阈值设置、LCD1602、超阈值报警、proteus) 功能描述: 基于STM32F103C8单片机实现的智能流速、流量,流量计设计 实现的功能是通过信号发生器模拟齿轮传感器,检测流量的大小,同时计算流过液体的总容量 可以设置最大流过的总容量,当超过设定值后通过蜂鸣器与LED灯指示 当没有超过则启动水泵控制电路带动液体流动 1、流速检测 2、流量统计 3、阈值显示与设置(通过按键实现阈值的调节或清零) 4、水泵启动 5、超阈值报警 有哪些资料: 1、仿真工程文件 2、PCB工程文件 3、原理图工程文件 4、源代码 ,核心关键词: 基于STM32的流量计; 智能流速流量监测; 水泵报警系统; 阈值设置; LCD1602; 超阈值报警; Proteus仿真; STM32F103C8单片机; 齿轮传感器; 信号发生器; 流量统计; 蜂鸣器与LED灯指示; 水泵控制电路。,基于STM32的智能流量监测与报警系统(阈值可调、流速与流量监
(灰度场景下的平面、海底、船、受害者)图像分类数据集【已标注,约1100张数据】 数据经过预处理,可以直接作为分类网络输入使用 分类个数【4】:平面、海底、船、受害者【具体查看json文件】 划分了训练集、测试集。存放各自的同一类数据图片。如果想可视化数据集,可以运行资源中的show脚本。 图像分类、分割网络改进:https://blog.csdn.net/qq_44886601/category_12858320.html 计算机视觉完整项目:https://blog.csdn.net/qq_44886601/category_12816068.html
arkime无geo下的oui文件
人脸识别项目实战
人脸识别项目实战
CAD 2025 二次开发dll
人脸识别项目源码实战
c语言学习
基于扩张状态观测器eso扰动补偿和权重因子调节的电流预测控制,相比传统方法,增加了参数鲁棒性 降低电流脉动,和误差 基于扩张状态观测器eso补偿的三矢量模型预测控制 ,基于扩张状态观测器; 扰动补偿; 权重因子调节; 电流预测控制; 参数鲁棒性; 电流脉动降低; 误差降低; 三矢量模型预测控制,基于鲁棒性增强和扰动补偿的电流预测控制方法
c语言学习