hibernate抓取问题
当使用xml配置类之间的关系时 ,例如 学生 班级,多对一关系
/**
* 默认情况会发出2条SQL语句,一条取student,一条取Classroom,其实这只需要一条sql
* 解决办法 通过设置XML中的<many-to-one name="classroom" column="cid" fetch="join"/> //默认fetch=select
* 可以完成对抓取的设置
*/
session = HibernateUtil.openSession();
Student stu = (Student)session.load(Student.class, 1); //只查一个学生
System.out.println(stu.getName()+","+stu.getClassroom().getName())
-----------------------------------------------------------------------------------------------------
缺点。有时候不需要班级信息也会查出来,占用内存 ,延迟加载就不起作用了
session = HibernateUtil.openSession();
Student stu = (Student)session.load(Student.class, 1);
//延迟加载就失效,自动查班级信息
System.out.println(stu.getName());
-----------------------------------------------------------------------------------------------------------------
但是当查询所有学生时 使用xml配置。并且使用fetch=join 如下代码 又会发查询学生sql和班级的sql
List<Student> stus = session.createQuery("from Student").list(); //查询全部学生
for(Student stu:stus) {
System.out.println(stu.getName()+","+stu.getClassroom().getName());
}
在XML中配置了fetch=join仅仅只是对load的对象有用,对HQL中查询的对象无用
解决办法1 。设置查询班级的数量 batch-size,但是查询的数据随着session关闭就没了,
<class name="Classroom" table="t_classroom" batch-size=''20''>
解决办法 2 使用fetch关键字 在hql中 ,如果使用了join fetch就无法使用count(*)
List<Student> stus = session.createQuery("select stu from " +
"Student stu join fetch stu.classroom").list();
for(Student stu:stus) {
System.out.println(stu.getName()+","+stu.getClassroom());
}
-------------------------------------------------------------------------------------------------------------------
当使用annotation配置类之间关系时。
----------------------------------------------------------------------------------------------------------
/**
* 对于Annotation的配置而言,默认就是基于join的抓取的,所以只会发出一条SQL
*/
session = HibernateUtil.openSession();
Student stu = (Student)session.load(Student.class, 1); //只查一个学生
System.out.println(stu.getName()+","+stu.getClassroom().getName());
-------------------------------------------------------------------------------------------------------
但是annotation默认没有延迟加载。如果查询全部学生。会自动查处关联的班级信息。发大量sql语句 这也是个问题
List<Student> stus = session.createQuery("from Student").list(); //查所有学生
for(Student stu:stus) {
System.out.println(stu.getName());
}
解决 使用annotation时设置ManToOne(fetch=FetchType.Lazy) 延迟加载 相当于xml配置中 默认也就是fetch='select'
影响 1 但是一下代码又会发2条sql
session = HibernateUtil.openSession();
Student stu = (Student)session.load(Student.class, 1);
System.out.println(stu.getName()+","+stu.getClassroom().getName())
影响 2
当又想查询所有学生和所在班级信息 如下代码,又会发多条SQL。查学生SQL。和班级SQL
List<Student> stus = session.createQuery("from Student").list();
for(Student stu:stus) {
System.out.println(stu.getName()+","+stu.getClassroom().getName());
}
------------------------------------------------------------------------------------------------------
@ManyToOne(fetch=FetchType.LAZY)//LAZY就是XML中的select,EAGER就表示XML中的join
annotation默认是EAGER
@BatchSize(size=20)
public class Classroom 相当于xml中的batch-size 设置
-------------------------------------------------
总结,当你使用xml时。默认有延迟。当查询一个对象时。同时也查他的关联对象,会发多条sql。先查对象,再发sql查关联对象。使用fetch=join配置,也是只对load对象有用(就是load方法里传的class),hql中的对象(hibernate自己发送的hql语句中查询的对象)没用还是发好几条sql.解决要不就设置batch-size要不就自己用 join fetch 自己写hql
使用annotation时,默认是没延迟。查询一个对象。只发一条。但是查询多个对象时 ,又和xml中使用fetch=join 的情况一下样 ,但是你设置fetcg=FetchType.LAZY ,查询单一对象时,又会发两条,所以 要不就设置batch-size要不就自己用 join fetch 自己写hql
所以有时候。用别人的东西。不了解别人的东西就会问题很多。但你花很多时间去研究他。不如花时间研究数据库,用原生 sql ,
懂数据库的人,可能不用hibernate, 想用好还得精通hibernate。
不懂数据库的人。估计也不会花心思研究hibernate。研究了hibernate。你还得懂数据库
hibernate抓取问题的更多相关文章
- Hibernate 抓取策略fetch-1 (select join subselect)
原文 :http://4045060.blog.51cto.com/4035060/1088025 部分参考:http://www.cnblogs.com/rongxh7/archive/2010/0 ...
- 029 hibernate抓取策略
实例A引用实例B,B如果是代理的话(比如多对一关联中):如果遍历A的查询结果集(假设有10条记录),在遍历A的时候,访问B变量,将会导致n次查询语句的发出!这个时候,如果在B一端的class上配置ba ...
- hibernate抓取策略
抓取策略(fetching strategy) 是指:当应用程序需要在(Hibernate实体对象图的)关联关系间进行导航的时候, Hibernate如何获取关联对象的策略.抓取策略可以在O/R映射的 ...
- Hibernate 抓取策略fetch-2 (批量抓取batch-size以及hibernate.jdbc.fetch_size、hibernate.jdbc.batch_size)
类关系: User N~1 Group 测试代码: System.out.println("1"); List stuList = session.createQuery(&quo ...
- Hibernate 抓取策略
抓取策略: 为了改变SQL语句执行的方式 当应用程序需要在Hibernate实体对象的关联关系间进行导航的时候,Hibernate如何获取关联对象的策略 抓取策略可以在O/R映射的元数据中声明,也可以 ...
- Hibernate批量抓取
------------------siwuxie095 Hibernate 批量抓取 以客户和联系人为例(一对多) 1.批量抓取 同时查询多个对象的关联对象,是 Hibernate 抓取策略的一种 ...
- Hibernate之加载策略(延迟加载与即时加载)和抓取策略(fetch)
假设现在有Book和Category两张表,表的关系为双向的一对多,表结构如下: 假设现在我想查询id为2的那本书的书名,使用session.get(...)方法: Session session=H ...
- 【Java EE 学习 48】【Hibernate学习第五天】【抓取策略】【二级缓存】【HQL】
一.抓取策略. 1.hibernate中提供了三种抓取策略. (1)连接抓取(Join Fetch):这种抓取方式是默认的抓取方式.使用这种抓取方式hibernate会在select中内连接的方式获取 ...
- Hibernate的抓取策略
立即检索:当执行某行代码的时候,马上发出SQL语句进行查询(get())延迟检索:当执行某行代码的时候,不会马上发出SQL语句进行查询.当真正使用这个对象的时候才会发送SQL语句(load()) 类级 ...
随机推荐
- C++ 内敛函数
在主调函数调用函数时,先将现场压入栈以保存现场-转去执行被掉函数-返回主调函数.现场出栈以恢复现场-继续往下执行. 为了减少函数调用的成本,特别是对于小型函数,C++提供了内敛函数(inline).C ...
- eclipse可以调试但是无法打开网页,提示一直在加载
工作过程中遇到了eclipse调试可以正常执行,也没有报错,但是通过浏览器打开网页就是打不开,还提示一直加载.这个问题找了很多网页和搜索引擎,大多数的方法就是重新配置elipse里的tomcat的we ...
- EDM模板制作规范
为了保证最大的兼容性,在制作HTML的email页面时,请严格按照规范来书写: 1.页面宽度推荐500px,最大不要超过750px: 2.制作HTML的email页面时,不使用css+div来布局,最 ...
- ASP.NET MVC 控制器通过继承控制器来达到 过滤 并且多了一个IAuthenticationFilter
暂时没有用到过这个IAuthenticationFilter接口,毕竟已经有三个具体实现类了,所以这个还不知道用在哪,以后看看 20190324 需要注意!!!控制器重写方法都是被protected修 ...
- python3.5在内存中获取远程图片尺寸
自已看看,用于获取选程图片的尺寸 >>> from io import BytesIO>>> import requests as rs>>> f ...
- 第一个.NET Core应用,创建.NET Core命令
打开cmd,依次输入mkdir .project(创建目录),cd .\.project(进入目录),dotnet new(新建初始项目),dotnet restore(还原依赖),dotnet ru ...
- mongodb 搭建集群(分片+副本集)
mongodb 搭建集群(分片+副本集) 一.搭建结构图: 二.搭建步骤:
- [NOI2010]能量采集(莫比乌斯反演)
题面: bzoj luogu NOI2010能量采集 题解 读完题之后我们发现在每个产生贡献的点\((x1,y1)\)中,它与原点之间的点\((x2,y2)\)都满足\(x2|x1\),\(y2|y1 ...
- leetcode 19. 删除链表的倒数第N个节点 JAVA
题目: 给定一个链表,删除链表的倒数第 n 个节点,并且返回链表的头结点. 示例: 给定一个链表: 1->2->3->4->5, 和 n = 2. 当删除了倒数第二个节点后,链 ...
- “全栈2019”Java第七十二章:静态内部类访问外部类成员
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...