三.通过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语句来做。
相关推荐
ORM框架如Hibernate,将Java对象与数据库中的表格进行映射,这样开发者就可以通过操纵对象而不是SQL语句来完成数据的存取,极大地提高了开发效率和代码的可维护性。 这份"hibernate学习笔记"涵盖了以下关键知识点:...
##### 知识点1:创建和操纵对象 - **创建对象**:`new 类型(参数列表)`。此语句创建了一个指定类型的对象实例。 - 示例:`String s = new String("Hello World");` - 其中,`new`是关键字,用于创建新的对象实例...
Hibernate是一种开源的对象关系映射(ORM)框架,它主要用于简化Java应用程序与关系数据库之间的交互。ORM框架的主要目标是消除直接的JDBC操作,通过提供一套高级API,使得开发者可以使用面向对象的方式来操作数据库...
linux基础进阶笔记,配套视频:https://www.bilibili.com/list/474327672?sid=4493093&spm_id_from=333.999.0.0&desc=1
IMG20241115211541.jpg
GEE训练教程——Landsat5、8和Sentinel-2、DEM和各2哦想指数下载
该资源内项目源码是个人的课程设计、毕业设计,代码都测试ok,都是运行成功后才上传资源,答辩评审平均分达到96分,放心下载使用! ## 项目备注 1、该资源内项目代码都经过严格测试运行成功才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),仅供学习参考, 切勿用于商业用途。
基于springboot家政预约平台源码数据库文档.zip
Ucharts添加stack和折线图line的混合图
基于springboot员工在线餐饮管理系统源码数据库文档.zip
新能源汽车进出口数据 1、时间跨度:2018-2020年 2、指标说明:包含如下指标的进出口数据:混合动力客车(10座及以上)、纯电动客车(10座及以上)、非插电式混合动力乘用车、插电式混合动力乘用车、纯电动乘用车 二、新能源汽车进出口月销售数据(分地区、分类型、分 级别) 1、数据来源:见资料内说明 2、时间跨度:2014年1月-2021年5月 4、指标说明: 包含如下指标 2015年1月-2021年5月新能源乘用车终端月度销量(分类型)部分内容如下: 新能源乘用车(单月值、累计值 )、插电式混合动力 月度销量合计(狭义乘用车轿车、SUV、MPV、交叉型乘用车); 月度销量同比增速(狭义乘用车轿车、SUV、MPV、交叉型乘用车); 累计销量合计(狭义乘用车轿车、SUV、IPV、交叉型乘用车); 累计销量同比增速(狭义乘用车轿车、SUV、MPV、交叉型乘用车); 累计结构变化(狭义乘用车轿车、SUV、IPV、交叉型乘用车); 2015年1月-2021年5月新能源乘用车终端月度销量(分地区)内容如下: 更多见资源内
中心主题-241121215200.pdf
内容概要:本文档提供了多个蓝奏云下载链接及其对应解压密码,帮助用户快速获取所需文件。 适合人群:需要从蓝奏云下载文件的互联网用户。 使用场景及目标:方便地记录并分享蓝奏云上文件的下载地址和密码,提高下载效率。 阅读建议:直接查看并使用提供的链接和密码即可。若遇到失效情况,请尝试联系上传者确认更新后的链接。
基于Java web 实现的仓库管理系统源码,适用于初学者了解Java web的开发过程以及仓库管理系统的实现。
资源名称:Python-文件重命名-自定义添加文字-重命名 类型:windows—exe可执行工具 环境:Windows10或以上系统 功能: 1、点击按钮 "源原文"【浏览】表示:选择重命名的文件夹 2、点击按钮 "保存文件夹"【浏览】表示:保存的路径(为了方便可选择保存在 源文件中 ) 3、功能①:在【头部】添加自定义文字 4、功能②:在【尾部】添加自定义文字 5、功能③:输入源字符 ;输入替换字符 可以将源文件中的字符替换自定义的 6、功能④:自动加上编号_1 _2 _3 优点: 1、非常快的速度! 2、已打包—双击即用!无需安装! 3、自带GUI界面方便使用!
JDK8安装包
配合作者 一同使用 作者地址没有次下载路径 https://blog.csdn.net/weixin_52372189/article/details/127471149?fromshare=blogdetail&sharetype=blogdetail&sharerId=127471149&sharerefer=PC&sharesource=weixin_45375332&sharefrom=from_link
GEE训练教程
该资源内项目源码是个人的课程设计、毕业设计,代码都测试ok,都是运行成功后才上传资源,答辩评审平均分达到96分,放心下载使用! ## 项目备注 1、该资源内项目代码都经过严格测试运行成功才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),仅供学习参考, 切勿用于商业用途。