hibernate lazy(延迟加载)

hibernate lazy策略可以使用在:
* 标签上,可以取值:true/false ,在hibernate3以上版本,默认是true
* 标签上,可以取值:true/false需要类增强工具
* 标签上,可以取值:true/false/extra
* 单端关联上,可以取值:false/proxy/no-proxy

lazy概念:只有真正使用该对象时,才会创建,对于hibernate而言,正真使用的时候才会发出sql

hibernate支持lazy策略只有在session打开状态下有效

1、在标签上,可以取值:true/false ,在hibernate3以上版本,默认是true

     默认为true,可不写,在执行查询语句时不进行,比如session.load(id)时,不执行sql语句(session.get(id)不支持lazy),而是在具体获取参数时,执行sql语句,比如obj.getName()。

    标签上的lazy特性只对普通属性起作用

    标签上的lazy不会影响到单端关联上的lazy特性

2、标签上,可以取值:true/false/extra
    lazy=true时,在获取对象的基本属性时,不进行set对象的查询,在获取set对象的具体属性时才进行查询,即在obj.getSet()不进行查询,obj.getSet().getName()时才进行查询

    lazy=false时,不延时,马上加载

    lazy=extra时,与=true时差不多(需要进一步验证

3、 单端关联上,可以取值:false/proxy/no-proxy

    默认为lazy=proxy, 在获取关联对象时,不进行查询,在获取关联对象的具体实现时,才执行sql查询

    lazy=false时,不延时,马上加载

     many-to-one 的lazy设为proxy,当child.getParent().getName()或child.getParent().f()时,parent都会被抓取,若设为no-proxy,调用child.getParent().f()时,parent是不会被抓取的,同时这种方式需要编译时字节码增强,否则和proxy没区别。

."编译时字节码增强" 是什么意思? 
"字节码增强"分编译期和运行期2种,编译期是修改java类编译后的class字节码文件,在上面附加“增强”操作

lazy (可选 - 默认为 proxy): 默认情况下,单点关联是经过代理的。lazy="no-proxy"指定此属性应该在实例变量第一次被访问时应该延迟抓取(fetche lazily)(需要运行时字节码的增强)。 lazy="false"指定此关联总是被预先抓取。注意,如果constrained="false", 不可能使用代理,Hibernate会采取预先抓取!

 

lazy="proxy",也是延迟加载的一种,需要满足以下条件;

比如说User和Group是many-to-one的关系,User中有一个类型为Group的属性group,   
  若你想要载入一个User时,它的Group属性group要延迟载入,   
    
  条件一:   必须为Group类定义proxy,这个proxy一般可以定义为两种   
      1   Group类本身   
      2   一个被Group类实现的,且包含了Group类的所有业务方法的接口,这种做法可以克服第一种做法的一些缺点,但会比较麻烦,因为要定义一个接口嘛。   
    
  条件二:   映射文件中这个的outer-join要定义为   
      1   "false"(如果你在hibernate.properties或hibernate.cfg.xml文件中定义了hibernate.use_outer_join   true)   
      2   "auto"(默认的)或"false"(hibernate.use_outer_join   false),(我没有试过当hibernate.use_outer_join   false时,的outer-join定义为"true",会不会延迟载入和还是会用outer   join一并取出关联对象(在这里就是group对象),你可以试试)   
    
  hibernate中proxy是用cglib生成的动态代理,相关源码可以看看相应的原码。   

 

引自: http://www.chinaovo.net/hibernate/253.htm

你可能感兴趣的:(hibernate lazy(延迟加载))