hibernate的延迟加载和抓取策略
一,延迟加载
1、实体类延迟加载 通过代理机制完成,由javassist类库实现运行时代理,修改实体类的字节码实现了运行时代理
<class lazy="true|false">
实体级别的延迟加载默认值为true,意味实体对象是延迟加载,只影响load方法。
<class lazy="true|false">其他查询方式都是立即加载
2、关联属性延迟加载 默认情况下除了<one-to-one>之外所有的关联属性都是延迟加载
A、单端关联对象<many-to-one lazy="proxy|no-proxy|false"> <one-to-one lazy="false|proxy|no-proxy">
通过javassist代理实现延迟加载
B、集合关联对象 <set lazy="true|extra|false">
在hibernate中为实现集合关联属性的延迟加载,没有使用jdk中的集合框架,它实现了一套自己的集合框架,
比如它的set集合通过PersistSet(该类实现java.util.Set接口)实现,
集合关联属性的lazy默认值为true,意味集合关联属性延迟加载。
lazy="extra"除指定集合属性延迟加载功能外,能够让hibernate智能感知开发者对关联的集合属性的操作类型,
hibernate会根据开发者对集合操作类型不同执行不同的sql语句,所以建议集合关联属性的lazy="extra"
Department dept = (Department) session.get(Department.class, 10); Set<Employee> set = dept.getEmps();//获取到关联的集合属性 //判断集合中元素个数是否为0 当关联属性lazy="true",如果我们仅仅需要判断集合中是否元素或者仅仅需要集合中元素个数不是这些元素
// select * from emp where deptno=?
boolean b = set.isEmpty();
//获取集合中元素的个数
int size = set.size();
Department dept = (Department) session.get(Department.class, 10); Set<Employee> set = dept.getEmps();//获取到关联的集合属性 //当lazy="extra"时,集合关联属性延迟加载,同时提供了一些功能,让hibernate能够智能感应开发者对集合关联属性的操作类型
//hibernate会根据开发者对集合属性的操作类型不同执行不同的sql语句
//当开发者执行调用集合属性的isEmpty()方法或者size()方法时发送select count(*) from emp where deptno=?
//当开发者执行遍历操作发送select * from emp where deptno=?
boolean b = set.isEmpty(); int size = set.size();
3、普通属性延迟加载 编译时对字节码进行修改
<property lazy="false|proxy|no-proxy">
关联对象的延迟加载有利于程序的运行效率,但在一些情况下会产生问题,如:延迟加载的关联对象在session关闭后如何获取?
- openSessionInView,把session的生命周期延长,保证在获取关联对象后关闭session;
- 在HQL语句中使用关键字fetch左外连接查询,立即获取主体对象和关联对象
List list=session.createQuery("from Employee e join fetch e.dept").list(); //Employee(dept)
3. 在关联属性映射信息中将fetch="join",会通过左外连接立即把关联对象查询出来(不建议使用)
4. 通过Hibernate工具类中initialize(Object lazyObject),立即加载关联对象
5. Criteria获取关联对象,调用setFetchMode("关联对象名",FetchMode m)将抓取方式m设置FetchMode.JOIN
二,抓取策略
关联对象的抓取方式的设置:
<many-to-one fetch="select|join">和<one-to-one fetch="join|select">
<set fetch="select|join|subselect">
join : 以左外连接形式立即获取关联对象(仅仅针对get方法和load方法时有效,对HQL查询、标准查询无效,不推荐设置)。
select :另外发送一条查询语句获取关联对象
subselect :另外发送一条查询语句或者子查询获取关联对象加载出来
三,1+n问题
什么是1+N问题:当我们发送1条sql语句获取主体对象的数据,假设获取到n个主体对象,
但是需要发送额外的n条数据获取关联对象。
解决1+n问题:
1、关联属性的映射文件信息中fetch="join",可以解决load和get方式的1+n问题(不建议)
2、使用HQL语句中关键字fetch,通过左外连接立即加载关联对象,所以该操作前提,我们确实需要立即将关联对象加载
3、使用Criteria查询语句,通过该对象的setFetchMode(String ass,FetchMode m) 设置为FetchMode.JOIN,
这样通过左外连接立即加载关联对象 只需要一条sql语句
4、设置批量加载,减少查询语句
1、在hibernate.cfg.xml中配置<property name="default.batch_fetch_size">2</property> 2、在集合关联属性<set batch-size="1|2|4|8|16">
单端的关联属性在单端对应的实体类<class batch-size="1|2|4|8|16">
5、将<set fetch="subselect">获取关联集合属性时通过子查询获取,避免1+N问题。
6、Query对象的iterator()方法引起的1+N问题要通过二级缓存解决
hibernate的延迟加载和抓取策略的更多相关文章
- 十、hibernate的延迟加载和抓取策略
延迟加载:控制sql语句发送时机 抓取策略:控制sql语句格式,子查询.连接查询.普通sql 延迟加载 延迟加载(lazy),也叫做懒加载:执行到该行代码时,不发送sql进行查询,只有在真正使用到这个 ...
- hibernate 延迟加载和抓取策略
一.延迟加载 1.简单查询get,load 针对对象本身延迟或即时 当使用load方法来得到一个对象时,此时hibernate会使用延迟加载的机制来加载这个对象,即:当我们使用session.load ...
- 【转】hibernate延迟加载和抓取策略
一.延迟加载 1.简单查询get,load 针对对象本身延迟或即时 当使用load方法来得到一个对象时,此时hibernate会使用延迟加载的机制来加载这个对象,即:当我们使用session.load ...
- Hibernate fetching strategies(抓取策略)
抓取策略(fetching strategies)是指:当应用程序需要在(Hibernate实体对象图的)关联关系间进行导航的时候,Hibernate如何获取关联对象的策略.抓取策略可以在O/R映射的 ...
- hibernate_06_hibernate的延迟加载和抓取策略
1.延迟加载 1>类级别的延迟加载 指的是通过oad方法查询某个对象的时候,是否采用延迟, session. load(Customer class1L) 类级别延迟加载通过<class& ...
- Hibernate 性能优化之抓取策略
fetch 抓取策略 前提条件:必须是一个对象操作其关联对象. 1. 根据一的一方加载多的一方,在一的一方集合中,有三个值:join/select/subselect 2.根据多的一方加载一的一方, ...
- Hibernate框架笔记04HQL_QBC查询详解_抓取策略优化机制
目录 1. Hibernate的查询方式 1.1 方式一:OID查询 1.2 方式二:对象导航查询 1.3 方式三:HQL方式 1.4 方式四:QBC查询 1.5 方式五:SQL查询 2. 环境搭建 ...
- Hibernate之加载策略(延迟加载与即时加载)和抓取策略(fetch)
假设现在有Book和Category两张表,表的关系为双向的一对多,表结构如下: 假设现在我想查询id为2的那本书的书名,使用session.get(...)方法: Session session=H ...
- 八 Hibernate延迟加载&抓取策略(优化)
面试:Hibernate效率很低,如何优化? 缓存怎么弄的,语句怎么优化? 聊聊一级缓存,聊聊抓取策略和延迟加载,聊聊批量抓取 延迟加载: 一般不单独使用,和延迟加载一起使用 延迟加载:lazy(懒加 ...
随机推荐
- linux安装扩展总结
---恢复内容开始--- 1.安装php 模块安装命令. wget http://pear.php.net/go-pear 执行 php go_pear 如果是php7 wget http://pea ...
- EditText格式化11位手机号输入xxx xxxx xxxx
EditText格式化11位手机号输入xxx xxxx xxxx /** * 格式化11位手机号码输入 xxx xxxx xxxx格式 * 如果一直是添加:输入到第三个或第8个数字时 ...
- tf.data
以往的TensorFLow模型数据的导入方法可以分为两个主要方法,一种是使用feed_dict另外一种是使用TensorFlow中的Queues.前者使用起来比较灵活,可以利用Python处理各种输入 ...
- vue 分享知识点
vue 分享模块清单 1.Vue 2.0之Vue实例和生命周期 2.vue 2.0之自定义指令 3.vue 2.0之观察者模式实现简单异步无限滚动 4.从JavaScript属性描述器剖析Vue.js ...
- redis实战笔记(10)-第10章 扩展Redis
本章主要内容 扩展读性能 扩展写性能以及内存容量 扩展复杂的查询 随着Redis的使用越来越多, 只使用一台Redis服务器没办法存储所有数据或者没办法处理所有读写请求的问题迟早都会出现, 这 ...
- 尝试用selenium+appium运行一个简单的demo报错:could not get xcode version. /Library/Developer/Info.plist doest not exist on disk
业余时间抽空搭了个appium+selenium的环境(mac), 在执行第一个脚本的时候遇到个问题纪录下: could not get xcode version. /Library/Develop ...
- 11.Proxy
Proxy Proxy 概述 Proxy 用于修改某些操作的默认行为,等同于在语言层面做出修改,所以属于一种"元编程"(meta programming),即对编程语言进行编程. ...
- c#基础学习(0703)之string.Format格式化日期
C# string.Format格式化日期 DateTime dt = ,,,,,,); string.Format("{0:y yy yyy yyyy}",dt); //17 1 ...
- MySQL填坑系列--Linux平台下MySQL区分大小写问题
大家好,我是软件大盗(道),下面开始我们的<MySQL填坑系列>. 笔者最近又在MySQL的边缘试探,然后,试探着,试探着就报错了. 书接上文,系统连接数据库时报错:找不到DB_TIMIN ...
- drawRect:和layoutSubview的区别
关于这两个方法的区别 还是有点意思的. UIView的setNeedsDisplay和setNeedsLayout方法.首先两个方法都是异步执行的.setNeedsDisplay会调用自动调用draw ...