一、Hibernate中的悲观锁(pessimistic)和乐观锁(optimistic)
1.hibernate悲观锁
1)hibernate自动设置悲观锁定
当使用Hibernate更新或者插入一行记录的时候,锁定级别自动设置为LockMode.WRITE。
当Hibernate在“可重复读”或者是“序列化”数据库隔离级别下读取数据的时候,锁定模式自动设置为LockMode.READ。这种模式也可以通过用户显式指定进行设置.
当事务结束,所有对象脱离锁定模式.
2)显式的用户指定悲观锁定方式
调用 Session.load()的时候指定锁定模式(LockMode)。
调用Session.lock()。
调用Query.setLockMode()。
当多个线程并发修改某条数据即对应的实体对象时,如果此时该对象正在被其中一个线程修改,那么,其他线程挂起,一直等待线程提交完成
如果在UPGRADE或者UPGRADE_NOWAIT锁定模式下调用Session.load(),并且要读取的对象尚未被session载入过,那么hibernate会发出sql语句载入对象。如果为一个session调用load()方法查询某个对象时,该对象已经在另一个较少限制的锁定模式下被载入了,那么Hibernate就对该对象调用lock()方法,即该对象被锁定。
session.load(Account.class, 1, LockMode.UPGRADE);
2.hibernate乐观锁
大多数基于数据版本记录机制(version)实现。一般是在数据库表中加入一个version字段
读取数据时,将版本号一同读出,之后更新数据时,版本号加1,如果提交的数据小于或者等于数据库表中的版本号,则认为数据是过时的,否则给予更新。
配置文件需要做如下改动,在<class>标签中加上optimistic-lock="version",在<class>内部<id>标签的下面定义<version ></version>
hibernate会自动维护version的版本号码,如果多个并发同时修改数据,先提交的会成功.
具体配置如下:
<class name="Account" table="tbl_account">
<id name="id" column="id">
<generator class="native"></generator>
</id>
<version name="version"></version>
<property name="name"></property>
<property name="balance"></property>
</class>
二、hibernate一级缓存
一级缓存生命周期很短和session的生命周期一致,一级缓存也叫session级缓存或事务级缓存
每个Session有自己的缓存,一级缓存只针对单个session,其他的session里面的缓存不会被查询到
哪些方法支持一级缓存:
* get()
* load()
* iterate(查询实体对象时),iterate查询普通属性,一级缓存不会缓存
Student student = (Student)session.createQuery("from Student s where s.id=1").iterate().next();
System.out.println("student.name=" + student.getName());
iterate会去数据库查ID,然后看这个ID的对象在一级缓存里面是否存在,如果存在的话就不会去数据库里面查
*list只会把对象放到缓存中,但不会从缓存中取对象
在同一个session中先save,在发出load查询save过的数据,不会查询数据库,sava支持缓存。
如何管理一级缓存:
* session.clear()
* session.evict()
如何避免一次性大量的实体数据入库导致内存溢出
* 分批导入强制先flush 然后再clear
如
for (int i=0; i<1000; i++) {
Student student = new Student();
student.setName("s_" + i);
session.save(student);
//每10条数据就强制session将数据持久化
//同时清除缓存,避免大量数据造成内存溢出
if ( i % 10 == 0) {
//flush是使数据持久化
session.flush();
//清空缓存clear,evict
//session.evict(Object object)
session.clear();
}
}
如果数据量特别大,考虑JDBC实现,如果JDBC也不能满足要求,可以考虑采用数据库本身的特定导入工具
三、hibernate二级缓存
二级缓存也称进程级的缓存,同时也称作sessionFactory级的缓存,二级缓存可以被同一个SessionFactory创建的所有session共享
二级缓存的生命周期和SessionFactory的生命周期一致,SessionFactory可以管理二级缓存
二级缓存的配置和使用:
默认情况下,Hibernate使用EHCache进行JVM级别的缓存
* 将ehcache.xml文件拷贝到src目录下,部分参数介绍
<ehcache>
<diskStore path="java.io.tmpdir"/>
<defaultCache
maxElementsInMemory="10000" <!-- 缓存最大的对象数目 -->
eternal="false" <!-- 缓存是否持久,不持久就会在规定时间内失效 -->
timeToIdleSeconds="120" <!-- 当缓存闲置n秒后销毁 -->
timeToLiveSeconds="120" <!-- 当缓存存活n秒后销毁-->
overflowToDisk="true" <!-- 是否保存到磁盘,当对象达到规定的最大数目时-->
/>
</ehcache>
* 开启二级缓存,修改hibernate.cfg.xml文件,加入如下配置
<property name="hibernate.cache.use_second_level_cache">true</property>
* 指定缓存产品提供商 修改hibernate.cfg.xml文件
<property name="hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider</property>
* 知道你过哪些实体类 使用二级缓存(两种方法)
* 在映射文件中采用<cache>标签,在class标签下配置<cache usage="read-only"/>必须配置到<id>标签的前面
* 在hiberanate.cfg.xml文件中,采用<class-cache>标签
<class-cache usage="read-only"/>
如何管理二级缓存:
* sessionFactory.evict(Student.class)从缓存中移除所有的Student类型的对象
* sessionFactory.evict(Student.class,id)从缓存中移除指定id的Student 类型的对象
二级缓存是缓存实体对象的
* 了解一级缓存和二级缓存的交互
CacheMode.NORMAL - 从二级缓存中读、写数据。默认
CacheMode.GET - 从二级缓存中读取数据,仅在数据更新时对二级缓存写数据。
CacheMode.PUT - 仅向二级缓存写数据,但不从二级缓存中读数据。
四、Hibernate查询缓存
查询缓存是针对普通属性结果集的缓存,对实体对象的结果集只缓存id(其ID不是对象的真正ID,它与查询的条件相关即where后的条件相关,不同的查询条件,其缓存的id也不一样)
查询缓存的生命周期,当前关联的表发生修改,或是查询条件改变时,那么查询缓存生命周期结束,它不受一级缓存和二级缓存 的生命周期的影响
查询缓存的配置和使用
* 在hiberante.cfg.xml文件中启用查询缓存,如:
<property name="hibernate.cache.use_query_cache">true</property>
* 在程序中必须手动启用查询缓存,如:
query.setCacheable(true);
启用了查询缓存query.list()就可以从查询缓存中取数据了,不用每次都重新发sql查询数据库了
分享到:
相关推荐
在企业级Java应用开发中,Spring和...总之,合理利用Hibernate的二级缓存机制,结合Spring的管理能力,可以有效地提升Java应用的性能。通过优化缓存配置和策略,可以在不牺牲数据一致性的情况下,达到良好的用户体验。
然而,由于对象模型和关系型数据库模型之间的映射会产生一定的性能开销,Hibernate引入了一套缓存机制来提高数据访问的效率。 #### 二、Hibernate的缓存结构 Hibernate的缓存机制分为两层:第一级缓存(Session...
在Java的持久化框架Hibernate中,缓存机制是提升系统性能的关键因素。它位于Hibernate应用和数据库之间,减少了对数据库的直接访问,从而提高了应用程序的运行速度。缓存中存储的是数据库数据的副本,当需要查询数据...
在处理大数据量或高并发的场景下,为了提高性能和减少数据库负载,Hibernate 提供了二级缓存机制。本文将深入探讨Hibernate的二级缓存,并结合相关源码进行解析。 ### 一、一级缓存与二级缓存 1. **一级缓存...
### Hibernate缓存机制详解 #### 一、Hibernate缓存机制原理及配置 **1.1 Hibernate缓存概述** Hibernate作为一款流行的Java持久层框架,其缓存机制是提高应用程序性能的关键之一。缓存的主要作用在于减少数据库...
2. Hibernate缓存机制.doc:这个文档应该详细解释了Hibernate的一级和二级缓存机制,包括缓存的生命周期、更新策略(如脏检查和锁机制)以及缓存失效策略。 3. osCache.doc:可能深入介绍了Oscache的高级特性和最佳...
缓存机制是提高ORM框架性能的核心,它减少了对数据库的直接访问,从而提升了数据读取速度。Hibernate提供了三级缓存结构: - **一级缓存**:每个Session都有一个本地缓存,即事务级缓存。在同一事务内,对同一对象的...
本文将深入探讨Hibernate的二级缓存机制,包括其工作原理、配置、优缺点以及常见问题。 ### 一、二级缓存的工作原理 一级缓存是Hibernate默认提供的Session级别的缓存,每次操作的数据都会被缓存到Session中,生命...
除此之外,文档还会介绍第二级缓存(Second Level Cache)和查询缓存(Query Cache),这两种缓存机制能显著提升应用程序的性能。Hibernate还支持JPA(Java Persistence API),因此,对于熟悉JPA的开发者,文档也会...
总结,Hibernate 的缓存机制是其性能优化的重要组成部分,理解和正确使用缓存能够显著提升应用效率,但也需要权衡缓存带来的潜在问题。在实际项目中,应根据业务需求和系统特性,灵活选择和配置缓存策略。
#### 六、缓存机制 - **一级缓存**: 默认开启,Session范围内的缓存,主要用于提高单次会话期间的性能。 - **二级缓存**: 需要手动配置,通常使用第三方缓存实现,如EHCache等,用于改善跨会话的查询性能。 #### 七...
2.4. Hibernate独有的注解扩展:除了标准的EJB3注解,Hibernate还提供了一些自定义注解,如`@GeneratedValue`用于设置主键生成策略,`@Formula`用于在属性中使用SQL表达式,`@Cache`用于配置缓存,`@Filter`用于动态...
同时,还介绍了Hibernate的一级缓存和二级缓存机制,并通过一个模拟银行转账程序的示例来展示了如何在实际开发中应用这些知识。通过学习这些内容,可以帮助开发者更好地理解Hibernate框架的核心特性,并在实际项目中...
10. **缓存机制**:解析第二级缓存和查询缓存的工作原理,以及如何配置和优化。 11. **事务与并发控制**:讨论Hibernate的事务管理,以及乐观锁、悲观锁等并发控制策略。 12. **事件监听与拦截器**:介绍如何使用...
8. **缓存机制**:Hibernate提供了两级缓存,一级缓存在SessionFactory级别,二级缓存可配置第三方缓存服务,如Ehcache或Infinispan,以提高性能和减少数据库负载。 9. **事务管理**:Hibernate支持JTA(Java ...
5. **缓存同步**:在多节点环境下,需要考虑如何保证缓存的一致性,比如使用分布式锁或版本号机制。 6. **性能调优**:监控缓存命中率,根据实际负载调整缓存大小和策略,避免缓存击穿或雪崩现象。 通过本教程,你...
在事务处理部分,手册会阐述Hibernate的事务管理机制,包括基于JTA和JDBC的事务控制,以及如何在服务层实现事务的正确性和一致性。同时,也会讨论乐观锁(Optimistic Locking)和悲观锁(Pessimistic Locking)的...
为了提高性能,Hibernate提供了两级缓存机制:一级缓存(session级别)和二级缓存(session factory级别),有效地减少了数据库访问次数。 ### 高级特性 - **懒加载(Lazy Loading)**:延迟加载关联对象,直到...