zoukankan      html  css  js  c++  java
  • 懒加载异常 途径

         首先,了解一下什么是延迟加载:

    所谓懒载入(lazy)就是延时载入。延迟载入。

    什么时候用懒载入呢,我仅仅能回答要用懒载入的时候就用懒载入。

    至于为什么要用懒载入呢,就是当我们要訪问的数据量过大时,明显用缓存不太合适,

    由于内存容量有限 ,为了降低并发量,降低系统资源的消耗。

    我们让数据在须要的时候才进行载入,这时我们就用到了懒载入。

    比方部门ENTITY和员工ENTITY,部门与员工1对多,假设lazy设置为 false。那么仅仅要载入了一个部门的po,就会依据一对多配置的关系把全部员工的po也载入出来。可是实际上有时候仅仅是须要用到部门的信息,不须要用到 员工的信息,这时员工po的载入就等于浪费资源。

    假设lazy设置为true,那么仅仅有当你訪问部门po的员工信息时候才回去载入员工的po的信息。

     

    hibernate3.0中lazy有三个值。true,false,proxy,默认的是lazy="proxy".
    详细设置成什么要看你的需求,并非说哪个设置就是最好的。
    假如在student对象中包括一个head对象
    假设你确定在用student对象的时候就要用到head对象里的属性。那你就设置马上载入,由于设置马上载入那么在查询student的同一时候就会查询 student的head。hibernate就会在查询的时候关联两张表从而生成的sql就可能仅仅有一条。

    而假设你设置的是延迟载入。那么肯定会要生成 1+N条sql语句:当中“1”是查询student的语句。“N”是依据N个student的id去查询head的N条语句。并且,延迟载入是要用到的 时候才去运行查询,这样系统推断那里须要载入,那里不须要载入也须要时间,性能上肯定就不如马上载入了!
    假设。你是有的地方须要用到student的时候才用到head属性,那么你就设置成延迟载入,由于查询2张表的数据肯定要比查询1张表的数据消耗大。
    到低要如何设置就要看你的实际需求了

    延迟载入机制是为了避免一些无谓的性能开销而提出来的。所谓延迟载入就是当在真正须要数据的时候,才真正运行数据载入操作。在Hibernate中提供了对实体对象的延迟载入以及对集合的延迟载入,另外在Hibernate3中还提供了对属性的延迟载入。


    A、实体对象的延迟载入 
    假设想对实体对象使用延迟载入,必需要在实体的映射配置文件里进行对应的配置。例如以下所看到的:
    <hibernate-mapping>
     <class name=”com.neusoft.entity.User” table=”user” lazy=”true”>
        ……
     </class>
    </hibernate-mapping>
         通过将class的lazy属性设置为true。来开启实体的延迟载入特性。

    假设我们执行以下的代码:
        User user=(User)session.load(User.class,”1”);(1)
        System.out.println(user.getName());(2)
      当执行到(1)处时,Hibernate并没有发起对数据的查询。假设此时通过一些调试工具。观察此时user对象的内存快照,会惊奇的发现,此时返 回的可能是User$EnhancerByCGLIB$$bede8986类型的对象。并且其属性为null,这是怎么回 事?session.load()方法会返回实体对象的代理类对象,这里所返回的对象类型就是User对象的代理类对象。在Hibernate中通过使用 CGLIB,来实现动态构造一个目标对象的代理类对象。并且在代理类对象中包括目标对象的全部属性和方法,并且全部属性均被赋值为null。通过调试器显 示的内存快照,能够看出此时真正的User对象,是包括在代理对象的CGLIB$CALBACK_0.target属性中。当代码执行到(2)处时。此时 调用user.getName()方法。这时通过CGLIB赋予的回调机制,实际上调用CGLIB$CALBACK_0.getName()方法,当调用 该方法时,Hibernate会首先检查CGLIB$CALBACK_0.target属性是否为null。假设不为空,则调用目标对象的getName 方法。假设为空。则会发起数据库查询,生成类似这种SQL语句:select * from user where id=’1’;来查询数据,并构造目标对象,而且将它赋值到CGLIB$CALBACK_0.target属性中。


      这样。通过一个中间代理对象,Hibernate实现了实体的延迟载入,仅仅有当用户真正发起获得实体对象属性的动作时。才真正会发起数据库查询操作。 所以实体的延迟载入是用通过中间代理类完毕的。所以仅仅有session.load()方法才会利用实体延迟载入。由于仅仅有session.load()方 法才会返回实体类的代理类对象。




    两种处理方法:
    一、这是延时载入的问题,把有关联的全部pojo类,在hibernate.cfg.xml文件里。

    一般在many-to-one中。set标签内中设lazy="false" 。

    二、用OpenSessionInViewFilter过滤器。注意hibernateFilter过滤器和struts2过滤器在映射时的先后顺序。

    同一时候要配置事物处理,否则会导致session处于仅仅读状态而不能做改动、删除的动作。

    在web.xml文件里增加一个filter例如以下:注意要载入 struts过滤器前面。

    <!-- 配置opensessionInViewFilter 用于解决懒载入问题-->
    	<filter>
    	   <filter-name>OpenSessionInViewFilter</filter-name>
    	   <filter-class>org.springframework.orm.hibernate3.support.OpenSessionInViewFilter</filter-class>
    	</filter>
    <filter-mapping>
    		<filter-name>OpenSessionInViewFilter</filter-name>
    		<url-pattern>*.action</url-pattern>
    </filter-mapping>


    分析原理:
    作用是拦截".action“结尾的请求,OpensessionInViewFilter的作用是将session延时关闭。
    Hibernate中懒载入错误的解决方法 - 九鬼 - 朝阳起又落
     
    在action类中会调用service层的方法,然后调用session.一般当session訪问结束之后就会将session关闭.
    比方我们要得到user的列表,默认情况下是不会将当中的roles的所有信息所有查出来的,最多仅仅给出一个id,或者是name.
    可是我们这个时候假设要打印roles的类容,就会报懒载入异常了.由于hibernate根本就没有将role的数据查出来.


    版权声明:本文博客原创文章,博客,未经同意,不得转载。

  • 相关阅读:
    接竹竿
    Vijos P1053 Easy SSSP
    计算机网络-五层协议和物理层
    代码阅读
    selenium自动化测试原理和设计的分享
    appium desktop 1.7 byName不能用,重写
    appium desktop 1.7 的swipe功能不能用,重写。
    appium在不同类中使用的是同一个session
    GIT 上传、ssh设置、一些命令。
    java 学习:在java中启动其他应用,由jenkins想到的
  • 原文地址:https://www.cnblogs.com/mfrbuaa/p/4632826.html
Copyright © 2011-2022 走看看