http://blog.csdn.net/yerenyuan_pku/article/details/70148567

Hibernate的一级缓存

Hibernate的一级缓存就是指Session缓存。通过查看Session接口的实现类——SessionImpl.java的源码可发现有如下两个类: 

  • actionQueue它是一个行动队列,它主要记录crud操作的相关信息。
  • persistenceContext它是持久化上下文,它其实才是真正的缓存。

在Session中定义了一系列的集合来存储数据,它们构成了Session的缓存。只要Session没有关闭,它就会一直存在。当我们通过Hibernate中的Session提供的一些API例如save()、get()、update()等进行操作时,就会将持久化对象保存到Session中,当下一次再去查询缓存中具有的对象(通过OID值来判断),就不会去从数据库中查询了,而是直接从缓存中获取。Hibernate的一级缓存存在的目的就是为了减少对数据库的访问。 
当然了,在Hibernate中还有一个二级缓存,它是SessionFactory级别缓存,我也不明白,所以在这儿,我就不做介绍了。

演示一级缓存的存在

现在我举例来演示一级缓存的存在。首先我们肯定要搭建好Hibernate的开发环境,读过我前面文章的童鞋,应该可以快速搭建好的,在此不做过多赘述。 
在cn.itheima.test包下新建一个单元测试类——HibernateTest.java,我们首先编写如下方法来测试一级缓存的存在:

public class HibernateTest {// 测试一级缓存@Testpublic void test3() { // 1.得到session Session session = HibernateUtils.openSession(); session.beginTransaction(); Customer customer = session.get(Customer.class, 1); // 查询id=1的Customer对象,如果查询到,会将Customer对象存储到一级缓存中 Customer customer2 = session.get(Customer.class, 1); // 会从一级缓存中查询,而不会向数据库再发送sql语句查询 // 2.事务提交,并关闭session session.getTransaction().commit(); session.close(); } }

首次查询id为1的Customer对象时,Hibernate会向MySQL数据库发送如下SQL语句:

selectcustomer0_.id as id1_0_0_,customer0_.name as name2_0_0_,customer0_.address as address3_0_0_, customer0_.sex as sex4_0_0_ from hibernateTest.t_customer customer0_ where customer0_.id=?

而第二次查询时,并没有向MySQL数据库发送select语句。这是因为首次查询id为1的Customer对象时,如果查询到,就会将Customer对象存储到一级缓存中,第二次查询时,会从一级缓存中查询,而不会向数据库再发送select语句查询。

持久化对象具有自动更新数据库的能力

现在我举例来演示持久化对象具有自动更新数据库的能力。在HibernateTest单元测试类中编写如下方法:

public class HibernateTest {// 持久化对象具有自动更新数据库的能力@Testpublic void test4() { // 1.得到session Session session = HibernateUtils.openSession(); session.beginTransaction(); Customer customer = session.get(Customer.class, 1); // 查询id=1的Customer对象,如果查询到,会将Customer对象存到一级缓存中 customer.setName("Tom"); // 操作持久化对象来修改属性 // 2.事务提交,并关闭session session.getTransaction().commit(); session.close(); } }

测试以上test4方法,将发现数据库t_customer表中,id为1的那条记录的name字段变为Tom,这足以说明持久化对象具有自动更新数据库的能力了。但是为什么持久化对象具有自动更新数据库的能力呢?原因涉及到一个概念——快照,快照就是当前一级缓存里面对象的散装数据(对象的属性,如name、id……)。当执行完以下这句代码:

Customer customer = session.get(Customer.class, 1);

就会向一级缓存中存储数据,一级缓存其底层使用了一个Map集合来存储,Map的key存储的是一级缓存对象,而value存储的是快照。通过在这句代码上打个断点,然后以debug的方式运行,Watch一下session会看得更加清楚,如下: 

接着执行以下这句代码:

customer.setName("Tom");

执行完毕会修改一级缓存中的数据,如下: 

当事务提交,session关闭,向数据库发送请求时,会判断一级缓存中数据是否与快照区一致,如果不一样,就会发送update语句。

一级缓存常用API

一级缓存特点:

  1. 当我们通过session的save、update、saveOrUpdate方法进行操作时,如果一级缓存中没有对象,那么会从数据库中查询到这些对象,并存储到一级缓存中。
  2. 当我们通过session的load、get、Query的list等方法进行操作时,会先判断一级缓存中是否存在数据,如果没有才会从数据库获取,并且将查询的数据存储到一级缓存中。
  3. 当调用session的close方法时,session缓存将清空。

一级缓存常用的API:

  1. clear():清空一级缓存。
  2. evict():清空一级缓存中指定的某个对象。
  3. refresh():重新查询数据库,用数据库中的信息来更新一级缓存与快照区。

现在我举例来演示一级缓存常用的API。在HibernateTest单元测试类中编写如下方法:

public class HibernateTest {// 测试一级缓存操作常用的API@Testpublic void test5() { // 1.得到session Session session = HibernateUtils.openSession(); session.beginTransaction(); // 操作 List<Customer> list = session.createQuery("from Customer").list(); // 这个操作会存储数据到一级缓存 session.clear(); // 清空一级缓存 Customer c = session.get(Customer.class, 1); // 会先从session的一级缓存中获取,如果不存在,才会从数据库里面获取 session.evict(c); // 从一级缓存中删除一个指定的对象 Customer cc = session.get(Customer.class, 1); cc.setName("kkkk"); session.refresh(cc); // refresh方法的作用是:它会用数据库里面的数据来同步我们的一级缓存以及快照区, // 这样的话,再操作cc时,就不会发送update语句。 // refresh方法:重新查询数据库,用数据库中的信息来更新一级缓存与快照区 // 2.事务提交,并关闭session session.getTransaction().commit(); session.close(); } }

可通过debug方式运行,这样会看得更清楚。在此不做过多赘述,读者自行操作。

Hibernate中常用API-Session的补充

讲完Hibernate持久化对象的三种状态和一级缓存之后,我就可以继续深入一点的讲解Session类中的以下方法了。

update

udpate操作主要是针对于脱管对象而言的,因为持久化对象具有自动更新数据库的能力。如果我们直接操作的对象是一个脱管对象,执行update会出现什么情况?如下:

public class HibernateTest {// session的update操作@Testpublic void test6() { // 1.得到session Session session = HibernateUtils.openSession(); session.beginTransaction(); // 执行update来操作一个脱管对象 Customer c = new Customer(); c.setAddress("天门"); c.setName("赵六"); c.setId(1); // 注意:我这里是为了模拟,所以手动给其赋值ID,正常的实际开发中是不建议这么做的 session.update(c); // 当执行update时,会将c放入到session的一级缓存 // 2.事务提交,并关闭session session.getTransaction().commit(); session.close(); } }

运行以上test6方法,可以发现数据库t_customer表中id为1的记录已经更新了。得出结论:update操作时,如果操作的对象是一个脱管对象,则可以操作,并且它会将脱管对象转换成持久对象再操作。但是这里依然会引发两个问题:

  1. 如果在session中出现相同的oid的两个对象,会产生如下异常:

    org.hibernate.NonUniqueObjectException: A different object with the same identifier value was already associated with the session : [cn.itheima.domain.Customer#1] at org.hibernate.engine.internal.StatefulPersistenceContext.checkUniqueness(StatefulPersistenceContext.java:648) at org.hibernate.event.internal.DefaultSaveOrUpdateEventListener.performUpdate(DefaultSaveOrUpdateEventListener.java:284) at org.hibernate.event.internal.DefaultSaveOrUpdateEventListener.entityIsDetached(DefaultSaveOrUpdateEventListener.java:227) at org.hibernate.event.internal.DefaultUpdateEventListener.performSaveOrUpdate(DefaultUpdateEventListener.java:38) at org.hibernate.event.internal.DefaultSaveOrUpdateEventListener.onSaveOrUpdate(DefaultSaveOrUpdateEventListener.java:73) at org.hibernate.internal.SessionImpl.fireUpdate(SessionImpl.java:703) at org.hibernate.internal.SessionImpl.update(SessionImpl.java:695) at org.hibernate.internal.SessionImpl.update(SessionImpl.java:690) at cn.itheima.test.HibernateTest.test6(HibernateTest.java:127) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source) at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source) at java.lang.reflect.Method.invoke(Unknown Source) at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50) at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12) at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47) at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17) at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325) at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78) at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57) at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290) at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71) at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288) at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58) at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268) at org.junit.runners.ParentRunner.run(ParentRunner.java:363) at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:86) at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:459) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:675) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:382) at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:192)

    示例代码为:

    public class HibernateTest {// session的update操作@Testpublic void test6() { // 1.得到session Session session = HibernateUtils.openSession(); session.beginTransaction(); // 得到id=1的Customer对象 Customer cc = session.get(Customer.class, 1); // session的一级缓存中存在了一个OID为1的Customer对象 // 执行update来操作一个脱管对象 Customer c = new Customer(); c.setAddress("天门"); c.setName("赵六"); c.setId(1); // 注意:我这里是为了模拟,所以手动给其赋值ID,正常的实际开发中是不建议这么做的 session.update(c); // 当执行update时,会将c放入到session的一级缓存 // 2.事务提交,并关闭session session.getTransaction().commit(); session.close(); } }

    运行以上方法就会报上面的那个异常。 
    结论:session的一级缓存里面是不能出现两个相同OID的对象的

  2. 脱管对象的oid如果在数据表中不存在,会报异常如下:

    org.hibernate.StaleStateException: Batch update returned unexpected row count from update [0]; actual row count: 0; expected: 1at org.hibernate.jdbc.Expectations$BasicExpectation.checkBatched(Expectations.java:67) at org.hibernate.jdbc.Expectations$BasicExpectation.verifyOutcome(Expectations.java:54) at org.hibernate.engine.jdbc.batch.internal.NonBatchingBatch.addToBatch(NonBatchingBatch.java:46) at org.hibernate.persister.entity.AbstractEntityPersister.update(AbstractEntityPersister.java:3071) at org.hibernate.persister.entity.AbstractEntityPersister.updateOrInsert(AbstractEntityPersister.java:2950) at org.hibernate.persister.entity.AbstractEntityPersister.update(AbstractEntityPersister.java:3330) at org.hibernate.action.internal.EntityUpdateAction.execute(EntityUpdateAction.java:145) at org.hibernate.engine.spi.ActionQueue.executeActions(ActionQueue.java:560) at org.hibernate.engine.spi.ActionQueue.executeActions(ActionQueue.java:434) at org.hibernate.event.internal.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:337) at org.hibernate.event.internal.DefaultFlushEventListener.onFlush(DefaultFlushEventListener.java:39) at org.hibernate.internal.SessionImpl.flush(SessionImpl.java:1282) at org.hibernate.internal.SessionImpl.managedFlush(SessionImpl.java:465) at org.hibernate.internal.SessionImpl.flushBeforeTransactionCompletion(SessionImpl.java:2963) at org.hibernate.internal.SessionImpl.beforeTransactionCompletion(SessionImpl.java:2339) at org.hibernate.engine.jdbc.internal.JdbcCoordinatorImpl.beforeTransactionCompletion(JdbcCoordinatorImpl.java:485) at org.hibernate.resource.transaction.backend.jdbc.internal.JdbcResourceLocalTransactionCoordinatorImpl.beforeCompletionCallback(JdbcResourceLocalTransactionCoordinatorImpl.java:147) at org.hibernate.resource.transaction.backend.jdbc.internal.JdbcResourceLocalTransactionCoordinatorImpl.access$100(JdbcResourceLocalTransactionCoordinatorImpl.java:38) at org.hibernate.resource.transaction.backend.jdbc.internal.JdbcResourceLocalTransactionCoordinatorImpl$TransactionDriverControlImpl.commit(JdbcResourceLocalTransactionCoordinatorImpl.java:231) at org.hibernate.engine.transaction.internal.TransactionImpl.commit(TransactionImpl.java:65) at cn.itheima.test.HibernateTest.test6(HibernateTest.java:130) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source) at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source) at java.lang.reflect.Method.invoke(Unknown Source) at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50) at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12) at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47) at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17) at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325) at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78) at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57) at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290) at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71) at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288) at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58) at org.junit

转载于:https://www.cnblogs.com/telwanggs/p/6963752.html

(转)Hibernate的一级缓存相关推荐

  1. Hibernate的一级缓存

    Hibernate的一级缓存 什么是缓存:缓存将数据库/硬盘上文件中数据,放入到缓存中(就是内存中一块空间).当再次使用的使用,可以直接从内存中获取 缓存的好处:提升程序运行的效率.缓存技术是Hibe ...

  2. Hibernate之一级缓存和二级缓存

    1:Hibernate的一级缓存: 1.1:使用一级缓存的目的是为了减少对数据库的访问次数,从而提升hibernate的执行效率:(当执行一次查询操作的时候,执行第二次查询操作,先检查缓存中是否有数据 ...

  3. hibernate教程--一级缓存详解

    1.1 Hibernate的一级缓存 1.1.1 Hibernate的一级缓存: 什么是缓存: 缓存将数据库/硬盘上文件中数据,放入到缓存中(就是内存中一块空间).当再次使用的使用,可以直接从内存中获 ...

  4. hibernate教程--一级缓存

    1.1Hibernate的一级缓存 1.1.1Hibernate的一级缓存: 什么是缓存: 缓存将数据库/硬盘上文件中数据,放入到缓存中(就是内存中一块空间).当再次使用的使用,可以直接从内存中获取. ...

  5. java中一级缓存_JavaWeb_(Hibernate框架)Hibernate中一级缓存

    Hibernate中一级缓存 Hibernate 中的缓存分为一级缓存和二级缓存,这两个级别的缓存都位于持久化层,并且存储的都是数据库数据的备份.其中一级缓存是 Hibernate 的内置缓存,在前面 ...

  6. hibernate之 一级缓存和二级缓存

    2019独角兽企业重金招聘Python工程师标准>>> 缓存 缓存的实现不仅需要作为物理介质的硬件,同时需要管理缓存的并发访问策略和过期策略的程序(软件).所以缓存通常是通过软件和硬 ...

  7. Hibernate学习—— 一级缓存快照

    上一篇说到,Hibernate框架中,对象转化为持久化状态后,会在事务提交时,自动将对象属性存储到数据库,而在事务提交之前,我们是对哪里的对象进行操作呢,这就是一级缓存(Session缓存)的作用. ...

  8. hibernate中的一级缓存

    hibernate中的一级缓存 Hibernate提供了两种缓存,这里主要写一级缓存. 第一级缓存是session的缓存,由于Session对象的生命周期通常对应一个数据事务或者一个应用事务,因此它的 ...

  9. hibernate一级缓存_Hibernate缓存–一级缓存

    hibernate一级缓存 Welcome to Hibernate Caching – First Level Cache Example Tutorial. Recently we looked ...

最新文章

  1. 原创:去繁存简,回归本源:微信小程序公开课信息分析《一》
  2. AS查看Android系统源码
  3. JS滚动条位置,顶部,底部,触发事件
  4. Liferay7 BPM门户开发之4: Activiti事件处理和监听Event handlers
  5. 反编译Android APK详细操作指南
  6. #并行优化# 容错算法 (Fault Tolerant)
  7. c++ int8_t转int_c专题之指针-----什么是指针?
  8. Flex读取txt文件里的内容(二)
  9. vue-cli目录结构解析
  10. 学好JAVA保终身_JAVA IO 学习
  11. HashTable、HashSet和Dictionary的区别
  12. 【转】飞鸽端口号被占用时的解决方法
  13. IAR在写结构体时不提示_U盘被写保护了以后怎么办?
  14. Python.密码本生成
  15. AR - 增强现实技术(AR)的103个应用场景汇总
  16. 外部中断控制LED灯开关
  17. ABAP -- 删除重复项
  18. 【linux kernel】基于ARM64分析linux内核的链接脚本vmlinux.lds.S
  19. 支持向量机(SVM)学习小记
  20. Python GUI编程之视频解析软件制作

热门文章

  1. ElasticSearch的中文分词
  2. RecordBatch分析
  3. mysql数据库单用户_SQLServer数据库之SqlServer数据库单用户模式无法删除的处理
  4. (73)Verilog HDL系统函数和任务:$monitor
  5. 条件随机场python实现_基于条件随机场的多标签分类
  6. 【Python】画图海龟
  7. 单片机RAM和ROM
  8. HTTP协议 (四) 缓存
  9. Python学习-将list列表写入文件并读取方法汇总
  10. C++中函数重载、缺省参数及命名空间