Hibernate第二级缓存是会话工厂的所有会话(Session)对象所使用的公共缓存。 如果您有来自会话工厂的多个会话(Session)对象,就可以操作会话工厂中的第二级缓存的数据。
SessionFactory类用于保存二级缓存数据。 它是所有会话对象的全局,默认情况下是不启用的。
不同厂商提供了二级缓存的实现。
1. EH二级缓存
2. OS二级缓存
3. Swarm二级缓存
4. JBoss二级缓存
每个实现提供不同的缓存使用功能。 有四种方法可以使用二级缓存。
1. 只读:缓存将适用于只读操作。
2. 非严格读写:缓存可用于读写,但一次只能读写。
3. 读写:缓存将用于读写,可以同时使用。
4. 事务处理:缓存将用于事务处理。
下面来看看看二级缓存实现和缓存使用情况。
使用EH缓存的二级缓存示例的额外步骤
1)在hibernate.cfg.xml文件中添加2个配置设置
<cache usage="read-only" />
3) 创建ehcache.xml文件
eternal 如果我们指定eternal =“true”,则不需要定义timeToIdleSeconds和timeToLiveSeconds属性,因为它将由hibernate内部处理。 指定eternal =“false”给程序员控制,但是我们需要定义timeToIdleSeconds和timeToLiveSeconds属性timeToIdleSeconds它定义了二级缓存中对象可以空闲多少秒。
timeToLiveSeconds它定义了在第二级缓存中对象可以存储多少秒,无论它是否空闲
SessionFactory类用于保存二级缓存数据。 它是所有会话对象的全局,默认情况下是不启用的。
不同厂商提供了二级缓存的实现。
1. EH二级缓存
2. OS二级缓存
3. Swarm二级缓存
4. JBoss二级缓存
每个实现提供不同的缓存使用功能。 有四种方法可以使用二级缓存。
1. 只读:缓存将适用于只读操作。
2. 非严格读写:缓存可用于读写,但一次只能读写。
3. 读写:缓存将用于读写,可以同时使用。
4. 事务处理:缓存将用于事务处理。
下面来看看看二级缓存实现和缓存使用情况。
实现 | 只读 | 非限制读写 | 读写 | 操作 |
EH二级缓存 | Yes | Yes | Yes | No |
OS二级缓存 | Yes | Yes | Yes | No |
Swarm二级缓存 | Yes | Yes | No | No |
JBoss二级缓存 | No | No | No | Yes |
使用EH缓存的二级缓存示例的额外步骤
1)在hibernate.cfg.xml文件中添加2个配置设置
<property name="hibernate.cache.use_query_cache">true</property>
<property name="cache.provider_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</property>
<property name="hibernate.cache.use_second_level_cache">true</property>
2) 缓存使用属性可以应用于hbm.xml文件中的类或集合级别,在hbm文件中添加缓存使用情况设置
<cache usage="read-only" />
3) 创建ehcache.xml文件
<?xml version="1.0"?>
<ehcache>
<defaultCache maxElementsInMemory="100" eternal="true"/>
</ehcache>
Hibernate二级缓存示例
项目的整体结构如下:
<?xml version='1.0' encoding='UTF-8'?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="demo.Student" table="student">
<cache usage="read-only"/>
<id name="id">
<generator class="increment"></generator>
</id>
<property name="name"></property>
</class>
hibernate.cfg.xml文件如下:
<?xml version='1.0' encoding='UTF-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
" http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
<property name="connection.driver_class">com.mysql.cj.jdbc.Driver</property>
<property name="connection.url">jdbc:mysql://localhost:3306/first_db?serverTimezone=UTC</property>
<property name="connection.username">root</property>
<property name="connection.password">9958</property>
<property name="dialect">org.hibernate.dialect.MySQLDialect</property>
<property name="show_sql">true</property>
<!-- 开启查询缓存 -->
<property name="hibernate.cache.use_query_cache">true</property>
<!-- 配置二级缓存 -->
<property name="hibernate.cache.use_second_level_cache">true</property>
<!--<property name="cache.provider_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</property>-->
<!--hibernate4以后改为如下-->
<property name="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</property>
<mapping resource="student.hbm.xml"/>
</session-factory>
</hibernate-configuration>
</hibernate-mapping>
ehcache.xml文件如下:
<?xml version="1.0" encoding="UTF-8"?>
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd" updateCheck="true"
monitoring="autodetect" dynamicConfig="true">
<diskStore path="java.io.tmpdir/ehcache" />
<defaultCache maxEntriesLocalHeap="10000" eternal="false"
timeToIdleSeconds="120" timeToLiveSeconds="120" diskSpoolBufferSizeMB="30"
maxEntriesLocalDisk="10000000" diskExpiryThreadIntervalSeconds="120"
memoryStoreEvictionPolicy="LRU" statistics="true">
<persistence strategy="localTempSwap" />
</defaultCache>
<cache name="org.hibernate.cache.internal.StandardQueryCache"
maxEntriesLocalHeap="5" eternal="false" timeToLiveSeconds="120">
<persistence strategy="localTempSwap" />
</cache>
<cache name="org.hibernate.cache.spi.UpdateTimestampsCache"
maxEntriesLocalHeap="5000" eternal="true">
<persistence strategy="localTempSwap" />
</cache>
<cache name="demo.Student" maxElementsInMemory="100"
eternal="false" timeToIdleSeconds="5" timeToLiveSeconds="200" />
</ehcache>
defaultCache将用于所有持久化类。 我们还可以通过使用 cache 元素来明确定义持久化类。eternal 如果我们指定eternal =“true”,则不需要定义timeToIdleSeconds和timeToLiveSeconds属性,因为它将由hibernate内部处理。 指定eternal =“false”给程序员控制,但是我们需要定义timeToIdleSeconds和timeToLiveSeconds属性timeToIdleSeconds它定义了二级缓存中对象可以空闲多少秒。
timeToLiveSeconds它定义了在第二级缓存中对象可以存储多少秒,无论它是否空闲
Test.java类如下:
public class Test {
public static void main(String[] args) {
// 在5.1.0版本汇总,hibernate采用如下新方式获取:
// 1. 配置类型安全的准服务注册类,这是当前应用的单例对象,不作修改,所以声明为final
// 在configure("cfg/hibernate.cfg.xml")方法中,如果不指定资源路径,默认在类路径下寻找名为hibernate.cfg.xml的文件
final StandardServiceRegistry registry = new StandardServiceRegistryBuilder()
.configure().build();
// 2. 根据服务注册类创建一个元数据资源集,同时构建元数据并生成应用一般唯一的的session工厂
SessionFactory sessionFactory = new MetadataSources(registry)
.buildMetadata().buildSessionFactory();
/**** 上面是配置准备,下面开始我们的数据库操作 ******/
Session session = sessionFactory.openSession();// 从会话工厂获取一个session
// creating transaction object
Transaction transaction = session.beginTransaction();
Statistics stats = sessionFactory.getStatistics();
System.out.println("Stats enabled="+stats.isStatisticsEnabled());
stats.setStatisticsEnabled(true);
System.out.println("Stats enabled="+stats.isStatisticsEnabled());
session.save(new Student(1, "tom"));
session.save(new Student(2, "jary"));
Session session1 = sessionFactory.openSession();
Student student = session1.load(Student.class, 1);
System.out.println(student.getId() + " >>>" + student.getName());
session1.close();
//再次查询ID=1的信息,因为使用了缓存,这里不会再发出查询语句...
Session session11 = sessionFactory.openSession();
Student student1 = session11.load(Student.class, 1);
System.out.println(student1.getId() + " " + student1.getName());
session11.close();
Session session2 = sessionFactory.openSession();
Student student2 = session2.load(Student.class, 2);
System.out.println(student2.getId() + " " + student2.getName());
session2.close();
transaction.commit();
session.close();
sessionFactory.close();
System.out.println("successfully saved");
}
}
运行结果如下:
可以看到,hibernate不会发出两次查询。 如果不使用二级缓存,hibernate将会发出两次查询,因为这两个查询都使用不同的会话对象。