当使用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抓取问题的更多相关文章

  1. Hibernate 抓取策略fetch-1 (select join subselect)

    原文 :http://4045060.blog.51cto.com/4035060/1088025 部分参考:http://www.cnblogs.com/rongxh7/archive/2010/0 ...

  2. 029 hibernate抓取策略

    实例A引用实例B,B如果是代理的话(比如多对一关联中):如果遍历A的查询结果集(假设有10条记录),在遍历A的时候,访问B变量,将会导致n次查询语句的发出!这个时候,如果在B一端的class上配置ba ...

  3. hibernate抓取策略

    抓取策略(fetching strategy) 是指:当应用程序需要在(Hibernate实体对象图的)关联关系间进行导航的时候, Hibernate如何获取关联对象的策略.抓取策略可以在O/R映射的 ...

  4. 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 ...

  5. Hibernate 抓取策略

    抓取策略: 为了改变SQL语句执行的方式 当应用程序需要在Hibernate实体对象的关联关系间进行导航的时候,Hibernate如何获取关联对象的策略 抓取策略可以在O/R映射的元数据中声明,也可以 ...

  6. Hibernate批量抓取

    ------------------siwuxie095 Hibernate 批量抓取 以客户和联系人为例(一对多) 1.批量抓取 同时查询多个对象的关联对象,是 Hibernate 抓取策略的一种 ...

  7. Hibernate之加载策略(延迟加载与即时加载)和抓取策略(fetch)

    假设现在有Book和Category两张表,表的关系为双向的一对多,表结构如下: 假设现在我想查询id为2的那本书的书名,使用session.get(...)方法: Session session=H ...

  8. 【Java EE 学习 48】【Hibernate学习第五天】【抓取策略】【二级缓存】【HQL】

    一.抓取策略. 1.hibernate中提供了三种抓取策略. (1)连接抓取(Join Fetch):这种抓取方式是默认的抓取方式.使用这种抓取方式hibernate会在select中内连接的方式获取 ...

  9. Hibernate的抓取策略

    立即检索:当执行某行代码的时候,马上发出SQL语句进行查询(get())延迟检索:当执行某行代码的时候,不会马上发出SQL语句进行查询.当真正使用这个对象的时候才会发送SQL语句(load()) 类级 ...

随机推荐

  1. LRU缓存介绍与实现 (Java)

    引子: 我们平时总会有一个电话本记录所有朋友的电话,但是,如果有朋友经常联系,那些朋友的电话号码不用翻电话本我们也能记住,但是,如果长时间没有联系 了,要再次联系那位朋友的时候,我们又不得不求助电话本 ...

  2. JavaScript函数参数问题

    声明一个回调函数 function(){ alert(data); } 这个时候,若调用这个函数的传入了一个同名data,则这里会引用到data的值,因为他们处于同一作用域.

  3. 文件查找记录类型 - TSearchRec - 文件操作(二)

    SysUtils单元下的TSearchRec是一个记录类型,主要通过FindFirst, FindNext, and FindClose使用. 接上一篇举例说明TSearchRec常用成员 //sys ...

  4. Reporting Service服务SharePoint集成模式安装配置(5、安装 SQL SERVER 2012 SP1产品)

    有过SQL2012 数据库安装经验的,可以跳过这一步骤直接进入第五步骤:RS外接程序的安装  数据库安装工具:SQLServer2012 SP1 Name:SQLServer2012SP1-FullS ...

  5. IIS虚拟目录加载NFS配置注意事项

    1,IIS下挂载的路径不要填写挂载的盘符,填 \\NFSIP地址\NFSID\ ,正确挂载的前提是在windows下开启了NFS客户端的功能. 2,IIS 网站中读写NFS 也不要用盘符,也用 步骤1 ...

  6. .net core MVC接受来自自前端的GET和POST请求方法的区别

    小伙伴们都知道,常用的http请求有两种方式,即GET方法和POST方法,很多刚入门的童鞋难免都会有一个误区,是不是GET是从服务器上获取数据,POST是向服务器传送数据? 我的理解是:不论是GET还 ...

  7. c# 求两个数中最大的值

    1.三元运算符: class Program { static void Main(string[] args) { ,); Console.WriteLine("最大数:{0}" ...

  8. Linux Shell 编程学习笔记

    1:Shell Script中if语句的条件部分要以分号来分隔 2:要注意条件测试部分中的空格.在方括号的两侧都有空格 3:echo "Hi, ${a}s" 单引号中的变量不会进行 ...

  9. AT2165 Median Pyramid Hard

    题目链接:戳我 一看范围1e5,往二分上想. 可是再怎么也没有想到这个神仙的二分答案qwq 我们二分一个数x,设比他大的数为1,小于等于他的数为0.那么我们就可以把原来的那个转化成一个01塔. 然后我 ...

  10. js验证汉字正则表达式

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...