`
format_me
  • 浏览: 31695 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

hibernate二级缓存

    博客分类:
  • java
阅读更多

整理自:http://www.iteye.com/topic/18904

 

Hibernate一级缓存是session级别的,默认开启的。如果在同一个session中对同一个id进行两次load()(get()不会使用缓存),Hibernate只会对数据库发起一次查询。

 

Hibernate二级缓存是全局缓存,默认为关闭状态。同时,它需要第三方缓存商提供实现。Hibernate3.5缓存策略提供商有EHCache、OSCache、JBoss Cache 1.x等。


Hibernate二级缓存配置:(以EHCache为例

1、hibernate.cfg.xml
<?xml version='1.0' encoding='UTF-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
          "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
          "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>

	<session-factory>
		......
		<property name="cache.provider_class">org.hibernate.cache.EhCacheProvider</property>
		<property name="hibernate.cache.use_query_cache">true</property>
		......
	</session-factory>

</hibernate-configuration>

2、EHCache配置文件ehcache.xml:(放在src目录下)
<?xml version="1.0" encoding="UTF-8"?>
<ehcache>
	<diskStore path="d:/ehcachedir/" /><!--缓存中的数据保存在磁盘上的路径-->
	<defaultCache maxElementsInMemory="1000" eternal="false"
		overflowToDisk="true" timeToIdleSeconds="300" timeToLiveSeconds="180"
		diskPersistent="false" diskExpiryThreadIntervalSeconds="120" />
</ehcache>
maxElementsInMemory为缓存中可以存放元素的个数,overflowToDisk属性的值为true,如果此时向缓存中添加第
1001个元素时,ehcache就会将缓存中的部分数据保存到磁盘上的临时文件中;eternal属性表示缓存是否永不超时,值为true表示永不超时;timeToLiveSeconds属性是缓存中每个元素(POJO)的超时时间,超过这个时间,该元素就会从缓存中被移走;timeToIdleSeconds是发呆时间,是可选的。

3、实体类映射配置文件:(*.hbm.xml)
<?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="com.charlie.bean.Zit" table="zit" catalog="bbc">

		<cache usage="read-only" /><!-- 设置缓存模式-->

		<id name="id" type="java.lang.String">
			<column name="id" length="7" />
			<generator class="assigned"></generator>
		</id>

		......
		
	</class>
</hibernate-mapping>
 4、Dao中的代码需要设置使用缓存:
public List findAllByCache() {
		log.debug("finding all Zit instances");

		List list = null;
		Session session = getSession();
		Transaction tx = null;
		String queryString = "from Zit";

		Query query = session.createQuery(queryString);
		query.setCacheable(true);//设置使用缓存
		try {
			tx = session.beginTransaction();
			list = query.list();
			tx.commit();
		} catch (RuntimeException re) {
			log.error("find all failed", re);
			tx.rollback();

		} finally {
			session.close();
		}
		return list;
}
 缓存我们可以简单的看成一个HashMap,通过key在缓存中查找value,对于数据库中的一条记录,也就是Java中的一个对象来讲,是根据id来找的,缓存的key就是id,value是pojo,当你调用load(),list(),iterate()方法,只要读出一个对象,都会写入缓存。iterate()比较特殊,它会先将符合条件的所有数据的id先查询出来,之后使用每个id进行load(),如果缓存中存在该对象就从缓存中取,如果不存在就直接查询数据库。执行iterate():
2012-02-23 21:23:45,439 DEBUG (AbstractBatcher.java:401) - 
    select
        zit0_.objuid as col_0_0_ 
    from
        bbc.zit zit0_
2012-02-23 21:23:45,486 DEBUG (AbstractBatcher.java:401) - 
    select
        zit0_.objuid as objuid0_0_,
        zit0_.atnm as atnm0_0_,
        zit0_.itat as itat0_0_,
        zit0_.itlv as itlv0_0_,
        zit0_.blat as blat0_0_,
        zit0_.importDate as importDate0_0_,
        zit0_.lend as lend0_0_ 
    from
        bbc.zit zit0_ 
    where
        zit0_.objuid=?
科目号:123 科目名称:短期信用贷款
2012-02-23 21:23:45,517 DEBUG (AbstractBatcher.java:401) - 
    select
        zit0_.objuid as objuid0_0_,
        zit0_.atnm as atnm0_0_,
        zit0_.itat as itat0_0_,
        zit0_.itlv as itlv0_0_,
        zit0_.blat as blat0_0_,
        zit0_.importDate as importDate0_0_,
        zit0_.lend as lend0_0_ 
    from
        bbc.zit zit0_ 
    where
        zit0_.objuid=?
科目号:12301 科目名称:工业短期信用贷款
2012-02-23 21:23:45,517 DEBUG (AbstractBatcher.java:401) - 
    select
        zit0_.objuid as objuid0_0_,
        zit0_.atnm as atnm0_0_,
        zit0_.itat as itat0_0_,
        zit0_.itlv as itlv0_0_,
        zit0_.blat as blat0_0_,
        zit0_.importDate as importDate0_0_,
        zit0_.lend as lend0_0_ 
    from
        bbc.zit zit0_ 
    where
        zit0_.objuid=?
科目号:12302 科目名称:商业短期信用贷款
 这样大家可能会想,同样的查询条件,第一次先list,第二次再iterate,就可以使用到缓存了。实际上这是很难的,因为你无法判断什么时候是第一次,而且每次查询的条件通常是不一样的,假如数据库里面有100条记录,id从1到100,第一次list的时候出了前50个id,第二次iterate的时候却查询到30至70号id,那么30-50是从缓存里面取的,51到70是从数据库取的,共发送1+20条sql。所以我一直认为iterate没有什么用,总是会有1+N的问题。

当某个ID通过hibernate修改时,hibernate会知道,于是移除缓存。 

使用二级缓存的前提:
你的hibernate程序对数据库有独占的写访问权,其他的进程更新了数据库,hibernate是不可能知道的。你操作数据库必需直接通过hibernate,如果你调用存储过程,或者自己使用jdbc更新数据库,hibernate也是不知道的。hibernate3.0的大批量更新和删除是不更新二级缓存的,但是据说3.1已经解决了这个问题。 
这个限制相当的棘手,有时候hibernate做批量更新、删除很慢,但是你却不能自己写jdbc来优化,很郁闷吧。 
SessionFactory也提供了移除缓存的方法,你一定要自己写一些JDBC的话,可以调用这些方法移除缓存,这些方法是: 
void evict(Class persistentClass) 
          //Evict all entries from the second-level cache. 
void evict(Class persistentClass, Serializable id) 
         // Evict an entry from the second-level cache. 
void evictCollection(String roleName) 
         // Evict all entries from the second-level cache. 
void evictCollection(String roleName, Serializable id) 
         // Evict an entry from the second-level cache. 
void evictQueries() 
         // Evict any query result sets cached in the default query cache region. 
void evictQueries(String cacheRegion) 
         // Evict any query result sets cached in the named query cache region. 
 
不过我不建议这样做,因为这样很难维护。比如你现在用JDBC批量更新了某个表,有3个查询缓存会用到这个表,用evictQueries(String cacheRegion)移除了3个查询缓存,然后用evict(Class persistentClass)移除了class缓存,看上去好像完整了。不过哪天你添加了一个相关查询缓存,可能会忘记更新这里的移除代码。如果你的jdbc代码到处都是,在你添加一个查询缓存的时候,还知道其他什么地方也要做相应的改动吗?


分享到:
评论

相关推荐

    Hibernate二级缓存

    Hibernate二级缓存是一种提高应用程序性能的技术,它将数据存储在SessionFactory级别的缓存中,使得数据可以在不同的Session之间共享。这与一级缓存(Session级别)不同,一级缓存仅存在于单个Session生命周期内,当...

    hibernate 二级缓存详解

    Hibernate 二级缓存是针对SessionFactory级别的全局缓存,与一级缓存(Session级别)不同,一级缓存只在单个Session生命周期内有效。二级缓存则允许不同Session之间共享数据,提高了数据访问效率,减少了对数据库的...

    Hibernate 二级缓存

    Hibernate 二级缓存

    hibernate二级缓存实例

    在这个"hibernate二级缓存实例"中,我们将深入探讨二级缓存的原理、配置以及在实际项目中的应用。 首先,我们需要了解一级缓存和二级缓存的区别。一级缓存是Session级别的,每个Session都有自己的一级缓存,用于...

    hibernate二级缓存示例源码

    **hibernate二级缓存详解** Hibernate作为Java领域中广泛使用的对象关系映射(ORM)框架,极大地简化了数据库操作。然而,在处理大量数据时,性能优化显得尤为重要,这就是二级缓存的作用。本文将深入探讨Hibernate...

    Hibernate 二级缓存 总结整理

    **Hibernate 二级缓存总结整理** 在Java的持久化框架中,Hibernate是一个广泛使用的ORM(对象关系映射)工具,它极大地简化了数据库操作。在处理大数据量或高并发的场景下,为了提高性能和减少数据库负载,...

    hibernate二级缓存包

    Hibernate二级缓存是Java开发中使用Hibernate框架进行数据持久化时优化性能的一种重要技术。它在一级缓存(Session级别的缓存)的基础上,提供了一个全局的、跨会话的数据存储层,可以显著减少对数据库的访问,从而...

    hibernate二级缓存java包下载

    二级缓存是 Hibernate 缓存策略的一部分,它在应用程序的多个会话之间共享数据,进一步优化了数据库访问效率。 二级缓存分为以下关键知识点: 1. **一级缓存与二级缓存的区别**: - 一级缓存:每个 Hibernate ...

    hibernate 二级缓存

    本篇文章将深入探讨Hibernate二级缓存的概念、工作原理以及如何在实际项目中设置和使用。 **一、二级缓存概念** 一级缓存是每个Hibernate Session内部的一个内存区域,用于存储Session期间的操作对象。当Session...

    hibernate二级缓存所需要的 jar包

    本篇将详细介绍Hibernate二级缓存的概念、作用以及所需jar包的作用。 一、Hibernate二级缓存概念 Hibernate的一级缓存是指Session级别的缓存,每个Session内部都有一个一级缓存,用于存储实体对象,当Session关闭时...

    day37 05-HIbernate二级缓存:一级缓存更新同步到二级缓存及二级缓存配置文件

    本篇文章将深入探讨Hibernate的二级缓存机制,以及如何进行一级缓存与二级缓存的同步,同时还会介绍二级缓存的配置文件设置。 一级缓存是Hibernate默认提供的缓存,每个SessionFactory实例都有一个一级缓存。当对象...

Global site tag (gtag.js) - Google Analytics