zoukankan      html  css  js  c++  java
  • Hibernate中get()和load()方法的区别

    在hibernate中我们知道如果要从数据库中得到一个对象,通常有两种方式,一种是通过session.get()方法,另一种就是通过session.load()方法,然后其实这两种方法在获得一个实体对象时是有区别的,在查询性能上两者是不同的。

    一.load加载方式

    当使用load方法来得到一个对象时,此时hibernate会使用延迟加载的机制来加载这个对象,即:当我们使用session.load()方法来加载一个对象时,此时并不会发出SQL语句,当前得到的这个对象其实是一个代理对象,这个代理对象只保存了实体对象的id值,只有当我们要使用这个对象,得到其它属性时,这个时候才会发出SQL语句,从数据库中去查询对象。

    session = sessionfactory.openSession();
    
    //通过load的方式加载对象时,会使用延迟加载机制,此时并不会发出sql语句,只有当我们需要使用的时候才会从数据库中去查询        
    Student s = (Student)session.load(Student.class, 2);

    我们看到,如果我们仅仅是通过load来加载我们的Student对象,此时控制台不会从数据库中查询出该对象,即并不会发出SQL语句。只有当我们要使用该对象时:

    session = sessionfactory.openSession();
        
    Student s = (Student)session.load(Student.class, 2);
    System.out.println(s);

    此时控制台会发出了sql查询语句,会将该对象从数据库中查询出来。

    其实第一次通过load获得的对象是一个代理对象,这个代理对象仅仅保存了id这个属性:

    session = sessionfactory.openSession();
                
    //通过load的方式加载对象时,会使用延迟加载机制,此时得到的对象其实是一个代理对象,该代理对象里面仅仅只有id这个属性
    //而且即使数据库中没有Id=10的字段,也不会报错
    Student s = (Student)session.load(Student.class, 10);
    System.out.println(s.getId());

    如果我们只打印出这个对象的id值时,此时控制台会打印出该id值,但是同样不会发出SQL语句去从数据库中去查询,即使数据库中没有Id=10的字段,也不会报错。这就印证了我们的这个Student对象仅仅是一个保存了id的代理对象,但如果我需要打印出Student对象的其他属性值时,这个时候会不会发出SQL语句呢?答案是肯定会发出的。

    那第二次执行load是怎么样的呢?

    session = sessionfactory.openSession();
                
    Student s1 = (Student)session.load(Student.class, 2);
    System.out.println(s1.getAge());
    Student s2 = (Student)session.load(Student.class, 2);
    System.out.println(s1==s2);//输出true

    下图显示了第一次执行Session的load()方法的过程。load方法先到Session缓存中查找OID为2的Student对象,由于Session缓存还不存在,于是load()创建一个OID为2的对象并返回,只有当要用到这个对象的时候,才到数据库中去查找,并将其保存在Session缓存中。

    下图显示了第二次执行Session的load()方法的过程。load()方法先到Session缓存中查找OID为2的Student对象,因为已经存在这样的Student对象了,就直接返回该Student对象。因此在上面的程序代码中,变量s1和变量s2引用的是同一个Student对象。

    二.get加载方式

    相对于load的延迟加载方式,get就直接的多,当我们使用session.get()方法来得到一个对象时,不管我们使不使用这个对象,此时都会发出SQL语句去从数据库中查询出来。

    session = sessionfactory.openSession();
                
    Student s1 = (Student)session.get(Student.class, 2);
    Student s2 = (Student)session.get(Student.class, 2);
    
    System.out.println(s1==s2);

    下图显示了第一次执行Session的get()方法的过程。get方法先到Session缓存中查找OID为2的Student对象,由于Session缓存还不存在,于是到数据库中去查找,并将其保存在Session缓存中。

    下图显示了第二次执行Session的get()方法的过程。get()方法第二次执行的过程与load()方法一致,先到Session缓存中查找OID为2的Student对象,因为已经存在这样的Student对象了,就直接返回该Student对象。因此在上面的程序代码中,变量s1和变量s2引用的也是同一个Student对象。

    三.使用get和load时的要注意的地方

    1、load()方法的这种加载方式称为懒加载,我们可以通过修改配置文件取消懒加载

    <class  name="Student" lazy="false" table="student">

    2、如果确定数据库中有这个对象就用load(),不确定就用get()(这样效率高)

    3、如果使用get方式来加载对象,当我们试图得到一个id不存在的对象时,此时会报NullPointException的异常。

    session = sessionfactory.openSession();
                
    Student s = (Student)session.get(Student.class, 10);        
    System.out.println(s.getName());    

    这是因为通过get方式我们会去数据库中查询出该对象,但是这个id值不存在,所以此时user对象是null,所以就会报NullPointException的异常了。

    4、如果使用load方式来加载对象,当我们试图得到一个id不存在的对象时,此时会报ObjectNotFoundException异常。

    session = sessionfactory.openSession();
                
    Student s = (Student)session.load(Student.class, 10);        
    System.out.println(s.getName());

    这是因为使用load时,此时的Student对象是一个代理对象,仅仅保存了当前的这个id值,当我们试图得到该对象的name属性时,这个属性其实是不存在的,所以就会报出ObjectNotFoundException这个异常了。

  • 相关阅读:
    git 从远程仓库指定分支克隆代码到本地
    vue路由懒加载
    ES6中拓展运算符 ...
    Mysql 安装-windows X64
    mysql-Federated存储方式,远程表,相当于sql server的linked server
    优化临时表使用,SQL语句性能提升100倍
    MySQL行锁深入研究
    mysql 队列 实现并发读
    mysql 常用sql
    mysql分表的3种方法
  • 原文地址:https://www.cnblogs.com/yzy-blogs/p/6743089.html
Copyright © 2011-2022 走看看