【Java EE 学习 47】【Hibernate学习第四天】【sesion】【一级缓存】【懒加载】

  一、Session概述

  1.Session 接口是 Hibernate 向应用程序提供的操纵对数据库的最主要的接口, 它提供了基本的保存, 更新, 删除和加载Java 对象的方法.

  2.理解Session的缓存

    使用缓存的目的:尽量减少访问数据库的频率

    (1)在 Session 接口的实现中包含一系列的 Java 集合, 这些 Java 集合构成了 Session 缓存. 只要 Session 实例没有结束生命周期, 存放在它缓存中的对象也不会结束生命周期

    (2)当session的save()方法持久化一个对象时,该对象被载入缓存,以后即使程序中不再引用该对象,只要缓存不清空,该对象仍然处于生命周期中。当试图load()对象时,会判断缓存中是否存在该对象,有则返回。没有在查询数据库

  3.清理缓存

     Session 具有一个缓存, 位于缓存中的对象称为持久化对象, 它和数据库中的相关记录对应. Session 能够在某些时间点, 按照缓存中对象的变化来执行相关的 SQL 语句, 来同步更新数据库, 这一过程被称为清理缓存(flush) 

  4.默认情况下 Session 在以下时间点清理缓存:

    (1)当应用程序调用 Transaction 的 commit()方法的时, 该方法先清理缓存(session.flush()),然后在向数据库提交事务(tx.commit())

    (2)当应用程序执行一些查询操作时,如果缓存中持久化对象的属性已经发生了变化,会先清理缓存,以保证查询结果能够反映持久化对象的最新状态

    (3)显式调用 Session 的 flush() 方法.

  5.flush/commit/refresh/clear四种方法的区别

    flush: 进行清理缓存(此时缓存中的数据并不丢失)的操作,让缓存和数据库同步 执行一些列sql语句,但不提交事务,;

       commit:先调用flush() 方法,然后提交事务. 则意味着提交事务意味着对数据库操作永久保存下来。

       reresh:刷新,让session和数据库同步,执行查询,把数据库的最新信息显示出来,更新本地缓存的对象状态.

       clear:清空缓存,等价于list.removeAll(); 

二、一级缓存

  1.与二级缓存不同的是,hibernate已经内部实现了一级缓存

  2.研究一级缓存需要研究的几个问题

    (1)一级缓存的生命周期:随着session的创建而被创建,随着session的关闭而消失

    (2)一级缓存是依赖于谁存在的:session对象

    (3)怎么样把数据存放到一级缓存中:使用get、load、save、update方法能够将数据存放到一级缓存中

    (4)怎么样从给一级缓存中获取数据:使用正常手段获取,如get、load,hibernate自动判断是否需要从一级缓存中获取数据

    (5)怎么样把缓存中的数据同步到数据库:使用flush方法

    (6)怎么样吧数据库中的数据同步到一级缓存中:使用refresh方法

    (7)一级缓存的特性

    (8)从一级缓存中清除某个对象:使用evict方法

    (9)清空一级缓存中所有的数据:使用clear方法

  3.测试一级缓存:https://github.com/kdyzm/day44_hibernate02

三、懒加载

  1.什么是懒加载:懒加载实际上就是“延迟加载”,这是属于优化策略的一种方法,当使用到的时候再加载是实现该技术的原理。

          最终目的:针对数据库中的大数据,不希望特别早的加载到内存中,当用到它的时候才加载

  2.懒加载策略分为三种:

    (1)类的懒加载

    (2)集合的懒加载

    (3)单端关联的懒加载

  3.类的懒加载

    (1)类的懒加载默认开启,在映射文件中的class标签上的lazy属性为true即开启了懒加载(默认为true),为false就关闭了懒加载。

    (2)必须使用load方法才能实现懒加载

    疑问:类的懒加载不能实现?使用get方法和使用load方法效果完全相同,使用debug模式可以看出来。

  4.集合的懒加载

    (1)针对一对多或者多对多的情况。 

    (2)在持久化类对应的映射文件中的set标签中设置lazy的属性值,可以取值true、false、extra三种取值,其中最好使用extra,使用该取值的时候能够实现最彻底的懒加载。也就是说使用true开启懒加载不如使用extra开启懒加载。

      最明显的例子就是使用一下代码测试的时候:

public void testTwo(){
        Session session=sessionFactory.openSession();
        
        Student student=(Student)session.get(Student.class,1L);
        Set<Course>courses=student.getCourses();
        System.out.println(courses.size());
        
        session.close();
    }

      如果lazy属性的值为true,虽然能够实现懒加载,但是在System.out.println(courses.size());语句执行的时候,发出的查询sql为:

Hibernate: select courses0_.sid as sid2_1_, courses0_.cid as cid1_, course1_.cid as cid0_0_, course1_.cname as cname0_0_ from course_stu courses0_ inner join test.course course1_ on courses0_.cid=course1_.cid where courses0_.sid=?

      也就是说,hibernate会将集合中的所有数据都从数据库中取出来,然而只是为了得到集合大小的值,并不需要这么做,所以可以改为extra,发出的sql语句为:

Hibernate: select count(cid) from course_stu where sid =?

  5.单端关联懒加载

    如果根据多的一方加载一的一方则由于数据量特别少,所以怎么样都好,所以开启懒加载或者不开启懒加载都无关紧要。

    lazy的属性值发生了变化:false、proxy、no-proxy,三种取值中false是不开启懒加载,proxy相当于true,开启懒加载。

  测试代码:在上面提供的代码中新建以下java文件即可测试。

 1 package com.kdyzm.hibernate.lazy;
 2 
 3 import java.util.Set;
 4 
 5 import org.hibernate.Session;
 6 import org.hibernate.SessionFactory;
 7 import org.hibernate.Transaction;
 8 import org.hibernate.cfg.Configuration;
 9 import org.junit.Test;
10 
11 import com.kdyzm.hibernate.domain.Course;
12 import com.kdyzm.hibernate.domain.Student;
13 
14 public class TestLazy {
15     private static SessionFactory sessionFactory;
16     static{
17         Configuration configuration=new Configuration();
18         configuration.configure();
19         sessionFactory=configuration.buildSessionFactory();
20     }
21      /*
22       * *    类的懒加载
23       *    集合的懒加载    
24       *    单端关联的懒加载
25       *    */
26     /*
27      * 类的懒加载
28      * 类的懒加载默认开启
29      * 必须使用load方法开启懒加载
30      */
31     @Test
32     public void testOne(){
33         Session session=sessionFactory.openSession();
34         Transaction transaction=session.beginTransaction();
35         
36         Student student=(Student)session.load(Student.class, 1L);
37         System.out.println(student.getSname());
38         
39         transaction.commit();
40         session.close();
41     }
42     /*
43      * 集合的懒加载
44      * 
45      * 这里针对的是关联对象的懒加载问题,即Set集合什么时候进行加载。
46      * 可以在set标签上设置lazy属性的值,如果为true则为懒加载;如果是false则关闭懒加载。
47      * 
48      * 集合的懒加载也是默认开启的
49      * 这里使用get方法和使用load方法效果完全相同?
50      * 在映射文件中配置set标签的lazy标签为false之后懒加载就失去效果了。
51      * 
52      * 在set标签上设置lazy属性的值为true还不如设置为extra,这样能够彻底的实现懒加载的特性。
53      */
54     @Test
55     public void testTwo(){
56         Session session=sessionFactory.openSession();
57         
58         Student student=(Student)session.get(Student.class,1L);
59         Set<Course>courses=student.getCourses();
60         System.out.println(courses.size());
61         
62         session.close();
63     }
64     /*
65      * 单端关联的懒加载
66      * 略。
67      */
68     @Test
69     public void testThree(){
70 
71         Session session=sessionFactory.openSession();
72         Transaction transaction=session.beginTransaction();
73         
74         transaction.commit();
75         session.close();
76     }
77 }
TestLayz.java

 

posted @ 2015-09-22 16:34  狂盗一枝梅  阅读(306)  评论(0编辑  收藏  举报