Hibernate深入学习(四):类级别的检索策略
来源:互联网 发布:云计算应用实例 编辑:程序博客网 时间:2024/06/05 21:09
hibernate提供了懒加载的检索策略,在没有用到类的相关属性时,不会发出sql从数据库中检索,而懒加载分为两种:类级别的懒加载和集合的懒加载,本章先来看看类级别的懒加载检索策略
以下是测试用的实体类:Parent,Children,他们是双向一对多的关系,类的具体属性如下:
//省略getter和setter,为了方便测试,请重写toString()public class Parent { private Integer id; private String name; Set<Children> childrens = new HashSet<>();}public class Children { private Integer id; private String name; private Parent parent;}
两个实体映射文件,为节约篇幅,省略property:
//Parent.hbm.xml<hibernate-mapping package="cn.sina.bean"> <class name="Parent" table="lazy_parent" lazy="true"> <set name="childrens" cascade="save-update" inverse="true"> <key column="pid"/> <one-to-many class="Children"/> </set> </class></hibernate-mapping>//Children.hbm.xml<hibernate-mapping package="cn.sina.bean"> <class name="Children" table="lazy_children"> <many-to-one name="parent" column="pid" class="Parent"/> </class></hibernate-mapping>
可以看到中将lazy=true,lazy默认的默认值就是true,我们知道get方法总是采用立即检索策略,所以想要看到效果就要用load方法,下面我们通过代码测试:
@Testpublic void testLazy(){ Parent parent = (Parent) session.load(Parent.class, 1); System.out.println(parent.getClass().getSimpleName());}
输出:Parent_$$_javassist_0
可以看到这是一个代理对象,并没有发出sql语句
此时我们增加一行代码:System.out.println(parent.getId());
得到结果:
Parent_$$_javassist_01
发现在没有发出sql的情况下获取到了ID,其实也很简单,id是我们传给load方法的,没必要走数据库
然后我们再加一行代码,获取name属性:System.out.println(parent.getName());
此时的输出为:
Parent_$$_javassist_01Hibernate: select parent0_.pid as pid1_0_, parent0_.name as name1_0_ from lazy_parent parent0_ where parent0_.pid=?tom
可以看到终于发出的sql语句,代理对象得到了初始化,不过请注意,本篇讨论的是类级别的检索策略,发出的sql并没有查询Parent的Set childrens = new HashSet<>(),这一点下一章再做讨论
懒加载异常
想象一种情况,在懒加载之后,我将session关闭了,关闭之后我还想获取到name属性,这时会怎么样?
@Testpublic void testLazy(){ Parent parent = (Parent) session.load(Parent.class, 1); System.out.println(parent.getClass().getSimpleName()); session.close(); //关闭session System.out.println(parent.getName());}
结果当然是抛异常了
org.hibernate.LazyInitializationException: could not initialize proxy - no Session
原因也写的很清楚,没session了
显式的立即检索
这里主要说一个Hibernate的静态方法:Hibernate.initialize(proxy);
看它的参数,接收的是一个代理对象,它的作用是在发生懒加载时,程序猿手动的加载
举例说明:
Parent parent = (Parent) session.load(Parent.class, 1);System.out.println(parent.getClass().getSimpleName());Hibernate.initialize(parent);
结果为:
Parent_$$_javassist_0Hibernate: select parent0_.pid as pid1_0_, parent0_.name as name1_0_ from lazy_parent parent0_ where parent0_.pid=?
同时Hibernate.isInitialized(proxy)可以返回一个boolean值,表示代理对象是否被初始化
类级别检索策略总结:
1.类级别的lazy属性默认为true
2.只要在获取类的非ID属性时,才会发出sql语句
3.发生懒加载时,想要在session关闭之后获取非ID属性会发生懒加载异常(LazyInitializationException)
4.Hibernate.initialize(proxy)可以显式的初始化代理对象
- Hibernate深入学习(四):类级别的检索策略
- hibernate类级别的检索策略
- Hibernate深入学习(五):集合的检索策略
- 类级别的检索策略
- Hibernate检索策略之类级别检索策略
- 【Hibernate学习笔记】Hibernate的检索策略
- hibernate的检索策略(抓取策略)
- Hibernate的检索策略
- Hibernate的检索策略
- Hibernate的检索策略
- Hibernate的检索策略
- Hibernate的检索策略
- hibernate的检索策略
- Hibernate的检索策略
- Hibernate的检索策略
- Hibernate的检索策略
- Hibernate的检索策略
- Hibernate的检索策略
- Linux指令学习(CentOs6.5)tail指令
- Swift-Charts 多种样式图表演示
- CGImageCreateWithImageInRect图片裁剪问题
- Android 数据存储(三)之数据库存储
- 应用系统之间数据传输的几种方式(转载)
- Hibernate深入学习(四):类级别的检索策略
- showModalDialog 4096限制,字符串参数突破方法
- shell脚本(一):变量的引用
- URL Scheme
- iOS 常写在pch中的设备区分
- sort uniq cut wc命令
- oracle常用函数
- 任务清单
- 图论算法