hibernate--lazy(懒加载)属性
关联映射文件中<class>标签中的lazy(懒加载)属性
Lazy(懒加载):只有在正真使用该对象时,才会创建这个对象
Hibernate中的lazy(懒加载):只有我们在正真使用时,它才会发出SQL语句,给我们去查询,如果不使用对象则不会发SQL语句进行查询。
Hibernate中lazy(懒加载)的实现:
采用了第三方组件的库,这个库叫cglib.jar(比较流行),这个库对我们的类生成代理类(JDK的动态代理,只能对JDK中实现了接口的类进行代理),代理可以控制源对象并且可以对源对象的功能进行增强,而cglib.jar可以对类进行代理(cglib对我们的类进行继承,生成一个子类,这个子类作为代理类返回给你)。
只有你正真代理类的方法,则会查看你有没有加载目标对象,如果没有则会加载目标对象。
Lazy(懒加载)在hibernate何处使用:
1、<class>标签上,可以取值:true/false,(默认值为:true)
2、<property>标签上,可以取值:true/false,需要类增强工具
3、<set>、<list>集合上,可以取值:true/false/extra,(默认值为:true)
4、<one-to-one>、<many-to-one>单端关联上,可以取值:false/proxy/noproxy
Session.load()方法支持lazy,而session.get()不支持lazy;
Hibernate的lazy生效期:
生效期和session一样的,session关闭,lazy失效
hibernate支持lazy策略只有在session打开状态下有效。
<class>标签上,可以取值:true/false,(默认值为:true):
实例一:设置<class标签上的lazy=true(默认)
@Test
public void LazyTest() {
session = HibernateUtil.getSession();
tx = session.beginTransaction(); // 执行此语并不会发SQL语句,只是返回一个代理类
Group group = (Group) session.load(Group.class, 1); // 不会发SQL语句,使用上面输入的1值
System.out.println("group.id=" + group.getId()); // 此处使用对象,会发出SQL语句
System.out.println("group.name=" + group.getName()); // 提交事务
tx.commit();
}
输出:
group.id=1
Hibernate: select group0_.id as id1_1_0_, group0_.name as name2_1_0_ from t_group group0_ where group0_.id=?
设置<class标签上的lazy=true(默认)
@Test
public void LazyTest1(){
Session session = null;
Transaction tx = null;
Group group = null; try {
session = HibernateUtil.getSession();
tx = session.beginTransaction(); group =(Group)session.load(Group.class, 1); System.out.println("group.name=" + group.getName()); //提交事务
tx.commit();
} catch (Exception e) {
e.printStackTrace();
tx.rollback();
} finally {
HibernateUtil.closeSession(session);
} //不能正确输出,抛出了LazyInitalizationException异常,因为session已经关闭
//hibernate支持lazy策略只有在session打开状态下有效。
System.out.println("group.name=" + group.getName());
}
关联映射文件中集合标签中的lazy(懒加载)属性
<set>、<list>集合上,可以取值:true/false/extra,(默认值为:true)
实例一:(集合上的lazy=true(默认))
@Test
public void LazyTest2() { session = HibernateUtil.getSession();
tx = session.beginTransaction();
// 不会发出SQL语句
Classes classes = (Classes) session.load(Classes.class, 1);
// 发出SQL语句,因为在使用对象
System.out.println("classes.name=" + classes.getName()); // 不会发SQL语句,只会返回一个代理类,因为没有使用对象
Set<Student> students = classes.getStudents(); // 会发出SQL语句,因为使用了对象
for (Iterator<Student> iter = students.iterator(); iter.hasNext();) {
Student student = iter.next();
System.out.println(student.getName());
} // 提交事务
tx.commit();
}
实例二:集合上的lazy=true(默认)
session = HibernateUtils.getSession();
tx = session.beginTransaction(); //不会发出SQL语句
Classes classes = (Classes)session.load(Classes.class, 1);
//发出SQL语句,因为在使用对象
System.out.println("classes.name=" + classes.getName()); //不会发SQL语句,只会返回一个代理类,因为没有使用对象
Set<Student> students = classes.getStudents(); //会发出SQL语句,发出查询全部数据的SQL,效率不高
System.out.println("student.count=" + students.size()); //提交事务
tx.commit();
实例三:集合上的lazy=false,其它保持默认
//不会发出SQL语句,因为只设置了集合上的lazy为false,其它保持默认
Classes classes = (Classes)session.load(Classes.class, 1);
//发出两条SQL语句,分别加载classes和student
//并且把集合中的数据也加载上来(虽然并没有使用集合中的对象),因为设置了集合的lazy=false
System.out.println("classes.name=" + classes.getName()); //不会发SQL语句,因为已经在前面加载了数据
Set<Student> students = classes.getStudents(); //会发出SQL语句,因为已经在前面加载了数据
for (Iterator<Student> iter = students.iterator();iter.hasNext();){
Student student = iter.next();
System.out.println(student.getName());
}
实例四:集合上的lazy=false,其它保持默认
//不会发出SQL语句
Classes classes = (Classes)session.load(Classes.class, 1);
//发出两条SQL语句,分别加载classes和student
//并且把集合中的数据也加载上来(虽然并没有使用集合中的对象),因为设置了集合的lazy=false
System.out.println("classes.name=" + classes.getName()); //不会发SQL语句,因为已经在前面加载了数据
Set<Student> students = classes.getStudents(); //不会发SQL语句,因为已经在前面加载了数据
System.out.println("student.count=" + students.size());
实例五:设置集合上lazy=extra,其它默认
session = HibernateUtils.getSession();
tx = session.beginTransaction(); //不会发出SQL语句
Classes classes = (Classes)session.load(Classes.class, 1); //会发出SQL语句
System.out.println("classes.name=" + classes.getName()); //不会发出SQL语句,只返回代理类
Set<Student> students = classes.getStudents(); //会发出SQL语句
for (Iterator<Student> iter = students.iterator();iter.hasNext();){
Student student = iter.next();
System.out.println(student.getName());
}
实例六:设置集合上lazy=extra,其它默认
session = HibernateUtils.getSession();
tx = session.beginTransaction(); //不会发出SQL语句
Classes classes = (Classes)session.load(Classes.class, 1);
//发出两条SQL语句
System.out.println("classes.name=" + classes.getName()); //不会发出SQL语句
Set<Student> students = classes.getStudents(); //发出SQL语句,发出一条比较智能的SQL语句(select count(id) form t_student where classesid=?)
System.out.println("student.count=" + students.size()); //提交事务
tx.commit();
<one-to-one>、<many-to-one>单端关联上的lazy(懒加载)属性
Ø <one-to-one>、<many-to-one>单端关联上,可以取值:false/proxy/noproxy(false/代理/不代理)
实例一:所有lazy属性默认(支持懒加载)
session = HibernateUtils.getSession();
tx = session.beginTransaction(); //不发出SQL语句,支持lazy(懒加载)
User user = (User) session.load(User.class, 3);
//发出SQL语句,只加载普通属性,集合中的数据不会加载
System.out.println("user.name=" + user.getName()); //不会发出SQL语句,只返回代理类
Group group = user.getGroup();
//发出SQL语句,因为现在真正使用对象
System.out.println("group.name=" + group.getName());
tx.commit();
实例二:将<many-to-one>中的lazy设置为false,其它默认
session = HibernateUtils.getSession();
tx = session.beginTransaction(); //不会发出SQL
User user = (User) session.load(User.class, 3);
//会发出SQL,发出两条SQL,分别是User和组
//因为<many-to-one>中的lazy=false,则会加载Group
System.out.println("user.name=" + user.getName()); //不会发出,已经在上面加载了数据
Group group = user.getGroup();
//不会发出,已经在上面加载了数据
System.out.println("group.name=" + group.getName());
tx.commit();
实例三:将<class>中的lazy设置为false,其它默认
session = HibernateUtils.getSession();
tx = session.beginTransaction(); //会发出SQL,因为<class>中的lazy=false
User user = (User) session.load(User.class, 3);
//不会发出SQL,已经在上面加载了
System.out.println("user.name=" + user.getName()); //不会发出,因为<class>标签上的lazy只对普通属性的影响
//<class>标签上的lazy不会影响到单端关联上的lazy特性
Group group = user.getGroup();
//会发出,因为开始使用对象
System.out.println("group.name=" + group.getName()); tx.commit();
hibernate--lazy(懒加载)属性的更多相关文章
- 018 关联映射文件中<class>标签中的lazy(懒加载)属性
Lazy(懒加载): 只有在正真使用该对象时,才会创建这个对象 Hibernate中的lazy(懒加载): 只有我们在正真使用时,它才会发出SQL语句,给我们去查询,如果不使用对象则不会发SQL语句进 ...
- 020 <one-to-one>、<many-to-one>单端关联上的lazy(懒加载)属性
<one-to-one>.<many-to-one>单端关联上,可以取值:false/proxy/noproxy(false/代理/不代理) 实例一:所有lazy属性默认(支持 ...
- 019 关联映射文件中集合标签中的lazy(懒加载)属性
<set>.<list>集合上,可以取值:true/false/extra,(默认值为:true) 实例一:(集合上的lazy=true(默认))class默认lazy=tru ...
- hibernate的懒加载问题
产生原因: 当使用hibernate查询一个对象的时候,如果Session关闭,再调用该对象关联的集合或者对象的时候,会产生懒加载异常! 解决方案: 方案一: 在Session关闭之前,查询对象关联的 ...
- Hibernate的懒加载session丢失解决方法
在web.xml加入spring提供的过滤器,延长session的生命周期 <!--Hibernate的懒加载session丢失解决方法 --> <filter> <fi ...
- hibernate 中 fetch=FetchType.LAZY 懒加载失败处理
对这种懒加载问题,最后的做法是利用Spring提供的一个针对Hibernate的一个支持类,其主要意思是在发起一个页面请求时打开Hibernate的Session,一直保持这个Session,使得Hi ...
- hibernate的懒加载
WHY? WHAT? HOW? 所谓懒加载(lazy)就是延时加载,延迟加载.即不是不加载,而是在需要的时候才加载. 什么时候用懒加载呢,我只能回答要用懒加载的时候就用懒加载. 至于为什么要用懒加载呢 ...
- Hibernate @OneToOne懒加载实现解决方案
在hibernate注解(三)中,我提高过一对一(@OneToOne)懒加载失效的问题.虽然给出了解决方法,但并没有给出完整的解决方案.今天我专门针对该问题进行讨论.至于懒加载失效的原因,在之前的文章 ...
- hibernate+spring mvc, 解决hibernate 对象懒加载 json序列化问题
引用地址 在使用Spring MVC时,@ResponseBody 注解的方法返回一个有懒加载对象的时候出现了异常,以登录为例: @RequestMapping("login") ...
随机推荐
- per-project basis
Of course, HSQLDB connection parameters should be stored on a per-project basis, instead of only onc ...
- ASP.NET MVC 学习第三天
今天来简单说一下Razor视图引擎语法相关的和视图类. 添加一个MvcTest项目,继续添加一个Home控制器,完成index的视图添加.我们就在index这里分析razor视图引擎.下面是home控 ...
- [原]项目进阶 之 持续构建环境搭建(三)Maven环境搭建
上次的博文项目进阶 之 持续构建环境搭建(二)Nexus私服器中,我们搭建了一个Nexus的maven私服,这次我们来重点讲解一下Maven的安装和配置.这里说明一下这次的环境搭建,比较基础,但却非常 ...
- Java Day 06
二维数组 定义: 格式1 int[][] arr = new int[3][2]; 格式2 int[][] arr = new int[3][];//每个一维数组初始化时为null 空指针异常 格式3 ...
- Zend Studio 12.0.2正式版发布和破解方法,zend studio 12.0.1汉化,相式设置为Dreamweaver,空格缩进为4个, 代码默认不折叠的设置,Outline中使用的图形标志,代码颜色之eot设置。
背景:zend studio 12.0.2 修复了一个12.0.1的: Fixed problem with referenced variables marked as undefined,我都说 ...
- 51nod 计算N!的位数
http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1130 对于这类问题:斯特林近似公式: 百度百科的证明:http: ...
- NodeJS - Express 3.0下ejs模板使用 partial展现 片段视图
如果你也在看Node.js开发指南,如果你也在一步一步实现 microBlog 项目!也许你会遇到本文提到的问题,如果你用的是Express 3.0 本书实例背景是 Express 2.0 而如今升级 ...
- python学习小结3:函数
Python是对接口编程,而不是对数据类型编程.例如我们定义了一个函数,在函数里用到了in这个接口,那么只要传入的参数实现了这个接口就可以,我们不在乎它是list还是tuple. 简单的函数 使用de ...
- Ubuntu系统下USB转串口的使用
PC系统是Ubuntu12.04,与路由器开发板之间用USB转串口线连接. 一.硬件连接 确认Ubuntu对USB转串口设备的支持. 1.# lsmod | grep usbserial如果有usbs ...
- Reactjs相比较原生方案是绝对的快吗?哪些情况下React有优势
作者:尤雨溪链接:http://www.zhihu.com/question/31809713/answer/53544875来源:知乎著作权归作者所有,转载请联系作者获得授权. 1. 原生 DO ...