刚才研究了一下预先抓取,有一点自己的看法。
首先:从定义上看,预先抓取是指hibernate通过select语句使用outer join(一般是left outer join)来获取对象的关联实例或者关联集合(重点是集合被初始化了)。但是预先抓取并非简单的通过外连接来取得数据。
这里有一个重点,假设有一个类A,有两个属性property1和property2 则HQL语句 from A as a left outer join a.property1 有一个原则--HQL语句忽略配置文件中的预先抓取策略。
这个策略对property1:不管配置文件中是什么策略(预先抓取、立即或延迟检索,它们都失效),这时都采用
HQL 指定的左外连接;而左外连接必定会初始化property1属性(或对象),但是如果配置文件里对property1
的检所策略是延迟加载,A类得不到对property1的引用,为了得到这个引用,需要hibernate再次发送一条SQL
来确定这种引用关系(在property1为集合时经常出现)。
这个策略对property2:配置文件对property2的预先抓取策略将被忽略(不管是fetch或是select),hibernate
只看立即和延迟加载(即由lazy="true"或是lazy="false"决定加载策略)
Hibernate管理这个策略的深度只是局限于由这个属性配置的对象
总结:1.仅从使用的角度讲,预先抓取和立即检索的效果一样,只是预先抓取可以减少SQL语句(如果设置hibernate.max_fetch_depth=0表示不允许使用外连接、即是使用立即加载策略。PS:一般不要设置这个数值过大以免造成过重的负担,影响性能)
2.预先抓取将初始化代理对象的引用,把对象的数据填充完毕,但是外连接仅把对象组装好,而不会初始化对象之间的引用关系(在集合引用时最为突出、想想只是外连接的话得到的仅是全是数据、并没有关联到对象之间引用)
3.注意HQL忽略的配置文件的预先抓取策略情况(上面提到的)
4.再给大家列出对象附属物的检索方式(对应下面的身份证,可以结合例子理解)
检索策略 | 使用HQL |
预先抓取,立即 | 预先抓取,保证抓取出来对象的关联 |
预先抓取,延迟 | 预先抓取,不保证抓取出来对象的关联 |
关于实际操作再来用班级、学生、身份证 来举个例子(一个班级可以有多个学生、一个学生只有一个身份证)
假设:学生对应身份证的策略是立即检索、预先抓取,
当要获得学生hibernate将会进行操作
Hibernate:select s.*,t.* from student s left outer join team t on s.team_id = t_id
Hibernate:select c.* from certificate c where c.id=?
Hibernate:select c.* from certificate c where c.id=?
......(重复查到的学生的个数)
如果策略是延迟检索、预先抓取
获得学生信息后,hibernate并不会立即对身份证信息加载(初始化),这时如果获取身份证的属性(或方法)会产生例外
学生对应班级,同理如果是立即检索
Hibernate:select c.*,s.* from student s inner join certificate c on s.id=c.id where s.team_id=?
如果是延迟检索也会出现例外。
以上说的是利用外连接即Hql为“from Student as s left join s.team”,如果用预先抓取的话即Hql为“from Student as s left join fetch s.team”关于策略上所要注意的是相同的,但是要注意:外连接获得的一条结果是一个对象组(学生、班级), 而预先抓取是一个对象(学生)
哈哈,忽然有所感悟:预先抓取与外联接就是获得的结果不一样(一个是一条一个学生,但是里面应该封装了所对应的班级;另一个是一条一组数据,给你学生和班级对象,这也许才是它的真谛!),书上说了这么多,都被绕晕了。。。。。。。。。
大家可以发表一下自己的看法 这个东西感觉很模糊,我这算是抛砖引玉了