hibernate一级缓存

注意:hql,sql语句不会使用一级缓存


一、Hibernate中的对象状态

  1.1、瞬时态(临时态)

    没有与Hibernate产生关联

    与数据库中的记录没有产生关联(有关联就是与数据库中表的id相对应)   

      获得:一般都只直接创建(new)
      瞬时态 转换 持久态
        一般操作:save方法、saveOrUpdate
      瞬时态 转换 脱管态
        一般操作:通过setId方法设置数据

   1.2、持久态

    Hibernate有关联

    对象有id

      获得:
        查询操作:get、loat、createQuery、createCriteria 等 获得都是持久态【】
        执行save之后持久态
        执行update之后持久态
      持久态 转换 瞬时态
        官方规定执行delete() --民间:删除态
      持久态 转换 脱管态
        session没有记录
        session.close () 关闭
        session.clear() 清除所有
        session.evict(obj) 清除指定的PO对象

  1.3、游离态(脱管态)

    没有与Hibernate产生关联

    对象有ID

      获得:
        创建、并设置OID的
        通过api获得
      脱管态 转换 瞬时态
        手动去除OID,设置成默认值
      脱管态 转换 持久态
        一般操作:update()、saveOrUpdate

  1.4、在代码中去查看三种状态

    

  1.5、三种状态的相互转换

    1.5.1、瞬时转化为持久

    

      注意:如果主键生成策略中value是assigned需要自己指定主键,不指定会报错。

    1.5.2、瞬时转化为游离  

        在数据库中id=1中已经存在有

        

        

    1.5.3、持久转化为瞬时

      

      方式二:

          

    1.5.4、持久转化为游离

      

    1.5.5、游离转化为瞬时

      

    1.5.6、游离转化为持久

      

  1.6、三种状态有什么用处?

    持久状态,我们使用Hibernate主要是为了持久化我们的数据.

    对于对象的状态,我们期望我们需要同步到数据库的数据,都被装换成持久状态

    持久化状态特点: Hibernate会自动将持久化状态对象的变化同步到数据库中.  

    

二、一级缓存

    又称为session级别的缓存。当获得一次会话(session),hibernate在session中创建多个集合(map),用于存放操作数据(PO对象),为程序优化服务,

    如果之后需要相应的数据,hibernate优先从session缓存中获取,如果有就使用;如果没有再查询数据库。当session关闭时,一级缓存销毁。

  在Hibernate中存在两种缓存:一是线程级别的缓存(session缓存)二是进程级别的缓存(二级缓存)

  缓存:用来提高效率的

    session缓存:就是session对象中存在的缓存,缓存中存在的是(持久化)对象。

  2.1、证明session缓存的存在

    

    我们用图形来理解:

      

  2.2、一级缓存快照

        在从数据库取数据时,会将数据一式两份,一份作为缓存中的对象,一份作为快照,在session提交时做对比。

      快照:与一级缓存一样的存放位置,对一级缓存数据备份。保证数据库的数据与 一级缓存的数据必须一致。如果一级缓存修改了,

          在执行commit提交时,将自动刷新一级缓存,执行update语句,将一级缓存的数据更新到数据库。

        

 

          

    举例:

      

    我们画图来分析一下: 

          

        

      持久化状态: 本质就是存在缓存中的对象,就是持久化状态.

  2.3、感受一级缓存带来的效率

      感受一:

    

      从上面这三行代码中,我们u1执行需要执行一条sql语句,我们执行得到u2和u3时,发现不会去再次查询,而是从缓存中找。

      感受二:

        

        从上面代码还在那个我们知道,第一个u1中会执行select语句。当事务还没有提交的时候,只会在缓存中进行。

        所以总的只会产生两条sql语句。

  2.4、一级缓存中的细节问题

    问题一:保存对象时,可以使用save方法和persist方法,他们之间有区别吗?

        没有区别,功能上是一样的。但persist(持久) 方法 来自于JPA 接口。而save(保存) 方法来自于Hibernate。

    问题二:HQL查询是否会使用一级缓存?

        HQL不会使用一级缓存.

    

    问题三:HQL语句批量查询时,查询结果是否会进入缓存?

        查询结果会放入缓存中

    

    问题四:SQL查询 结果会不会放入1级缓存中?

        如果把查询结果封装到对象中,对象会放入一级缓存

      

    问题五:SQL查询 结果会不会放入1级缓存中?

        没有把查询结果封装到对象中,对象不会放入一级缓存

          

    问题六:criteria  会将查询结果放入一级缓存. 但是查询不会使用一级缓存. 与Hql查询结论一致.    

三、session中的其他API

    evict()方法:将缓存中的对象删除

    clear()方法:清空一级缓存

    refresh方法:刷新,强制刷新缓存中的对象 => (可以用来解决缓存与数据库数据不同步的问题)

    

    flush方法:对比快照,并提交缓存对象


posted @ 2018-03-07 20:09  車輪の唄  阅读(10)  评论(0编辑  收藏  举报  来源