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

使用Spring3.1 Cache注解+EHCache遇到的问题

 
阅读更多

关于Spring3.1 Cache注解的介绍请参看http://hanqunfeng.iteye.com/blog/1158824

 

这里只说一下遇到的问题,web应用,session超时为30分钟,使用Cache注解+EHCache,如下:

 
<cache name="andCache" maxElementsInMemory="10000"  
        maxElementsOnDisk="1000" eternal="false" overflowToDisk="true"  
        diskSpoolBufferSizeMB="20" timeToIdleSeconds="300" timeToLiveSeconds="600"  
        memoryStoreEvictionPolicy="LFU" />  

说明:缓存名andCache,内存中最多可缓存10000个Element,其中的element会在闲置5分钟或是存活10分钟之后失效,

超过10000element时,element将会输出到磁盘中。

 

结果,在登录系统后,如果超过5分钟不访问系统,则会出现异常,

按照这样配置,我原以为,如果5分钟之内不访问应用,则缓存会失效,那么spring就会重新执行原先被缓存的方法,但实际上,方法并没有被重新执行,而是返回了null,貌似spring依旧是从缓存中查找,并且找到了key,只不过value已经是null了。

 

这样就会导致系统出现空指针异常,况且session也尚未超时,

但就算是将闲置时间设置为30分钟以上,用户在下次登录时一样会出现异常,但是第二次登录就又会正常,然后周而复始,貌似,只有当第一次取到null后,spring才会重新执行方法。

 

感觉上,好像ehcache清除缓存后,spring并不知道,因为key依旧可以在spring的cache中找到,但是通过key取值时,spring会从ehcache中查找,由于ehcache已经清除了缓存,所以返回值一定是null,而spring也把这个null当成了被缓存的value返回给系统。

 

使用spring自带的缓存机制没有遇到该问题。

 

分享到:
评论
5 楼 xiaoyaozjl 2013-04-08  
LZ使用的是spring 3.1.0.M1版本,也就是里程碑版本非正式版,是存在这个问题的。在spring 3.1.0.M2里面已经修正了。

3.1.0.M1版本的org.springframework.cache.ehcache.EhCacheCache.java:
public Object get(Object key) {
	Element element = cache.get(key);
	return (element != null ? element.getObjectValue() : null);
}


3.1.0.M2版本的org.springframework.cache.ehcache.EhCacheCache.java:
public ValueWrapper get(Object key) {
	Element element = cache.get(key);
	return (element != null ? new DefaultValueWrapper<Object>(element.getObjectValue()) : null);
}
4 楼 yaofaye 2013-02-26  
对的,返回的null是,key对应的value值为null,并不是Cache.ValueWrapper为null,客户端判断一下value值,如果为null,返回一个null的Cache.ValueWrapper即可。
songbin0201 写道
引用
但实际上,方法并没有被重新执行,而是返回了null,貌似spring依旧是从缓存中查找,并且找到了key,只不过value已经是null了。


从源码上看不应该是这样吧?
代理方法会拦截并判断Cache.ValueWrapper是否为空,如果为空直接invoke原方法,返回原方法中查询出来的对象

// follow up with cacheable
//inspectCacheables根据cachename验证Cache.ValueWrapper是否为空
CacheStatus status = inspectCacheables(ops.get(CACHEABLE));

Map<CacheOperationContext, Object> updates = inspectCacheUpdates(ops.get(UPDATE));

if (status != null) {
if (status.updateRequired) {
   updates.putAll(status.cUpdates);
}
// return cached object
else {
   return status.retVal;
}
}

retVal = invoker.invoke();//retVal如果为空,直接调用原方法,返回非缓存数据



inspectCacheables方法中关于验证Cache.ValueWrapper的代码,为空设置localCacheHit 为false,localCacheHit为false,设置需要重新更新cache

boolean localCacheHit = false;

					// check whether the cache needs to be inspected or not (the method will be invoked anyway)
					if (!updateRequire) {
						for (Cache cache : context.getCaches()) {
							Cache.ValueWrapper wrapper = cache.get(key);
							if (wrapper != null) {
								retVal = wrapper.get();
								localCacheHit = true;
								break;
							}
						}
					}

					if (!localCacheHit) {
						updateRequire = true;
					}

3 楼 songbin0201 2012-12-17  
引用
但实际上,方法并没有被重新执行,而是返回了null,貌似spring依旧是从缓存中查找,并且找到了key,只不过value已经是null了。


从源码上看不应该是这样吧?
代理方法会拦截并判断Cache.ValueWrapper是否为空,如果为空直接invoke原方法,返回原方法中查询出来的对象

// follow up with cacheable
//inspectCacheables根据cachename验证Cache.ValueWrapper是否为空
CacheStatus status = inspectCacheables(ops.get(CACHEABLE));

Map<CacheOperationContext, Object> updates = inspectCacheUpdates(ops.get(UPDATE));

if (status != null) {
if (status.updateRequired) {
   updates.putAll(status.cUpdates);
}
// return cached object
else {
   return status.retVal;
}
}

retVal = invoker.invoke();//retVal如果为空,直接调用原方法,返回非缓存数据



inspectCacheables方法中关于验证Cache.ValueWrapper的代码,为空设置localCacheHit 为false,localCacheHit为false,设置需要重新更新cache

boolean localCacheHit = false;

					// check whether the cache needs to be inspected or not (the method will be invoked anyway)
					if (!updateRequire) {
						for (Cache cache : context.getCaches()) {
							Cache.ValueWrapper wrapper = cache.get(key);
							if (wrapper != null) {
								retVal = wrapper.get();
								localCacheHit = true;
								break;
							}
						}
					}

					if (!localCacheHit) {
						updateRequire = true;
					}

2 楼 jahu 2012-12-15  
登录用户,不用缓存吧,
 
1 楼 snowolf 2012-09-24  
多个应用,分布式部署,缓存数据如何保持一致性?

相关推荐

    Spring 3.1 Cache Abstraction Tutorial

    Spring 3.1的缓存抽象提供了一种统一的方式来管理和使用缓存。它不仅支持简单的缓存存储,如 EhCache 和 Hazelcast,还允许与复杂的分布式缓存系统如 Redis 和 Guava 集成。通过注解驱动的编程模型,开发者可以在...

    Spring+EhCache缓存实例

    Spring框架自3.1版本开始引入了统一的缓存抽象,支持多种缓存实现,其中包括EhCache。Spring的缓存抽象层使得切换缓存提供商变得简单,而无需改动大量代码。 **3. 配置Spring与EhCache** 在Spring中使用EhCache,...

    Ehcache(一): Spring + Ehcache开场白

    Spring框架自3.1版本起引入了抽象的缓存管理机制,允许开发者选择不同的缓存实现,如Ehcache、Guava Cache等。通过声明式或编程式的缓存注解,可以轻松地在方法级别启用缓存功能。 集成Ehcache,首先需要在项目中...

    spring 3.1中的cache小结

    例如,使用`&lt;cache:advice&gt;`和`&lt;cache:annotation-driven&gt;`元素可以关联缓存注解与具体的缓存策略。 6. **缓存异常处理**:Spring提供了一套完整的异常处理机制,当缓存操作失败时,可以配置异常处理器来决定如何...

    Spring与ehcache结合使用

    ### Spring与ehcache结合使用详解 #### 一、前言 在现代软件开发中,缓存技术被广泛应用于提高应用程序的性能。其中,Spring框架因其灵活性和强大的功能,在Java领域得到了广泛应用;而ehcache作为一款高性能、...

    spring3.1 api

    新增了Cache抽象,提供了一种统一的缓存管理接口,支持Ehcache、Guava等主流缓存库,方便实现高效的缓存策略。 7. **表达式语言(SpEL)** Spring Expression Language在3.1版本中增强了其功能,支持更多的操作符...

    Spring基于注解的缓存配置--EHCache AND OSCache

    本篇文章将深入探讨如何使用注解配置Spring与EHCache或OSCache这两个流行的Java缓存解决方案。以下是对该主题的详细阐述: 1. **Spring缓存抽象** Spring 3.1引入了一种统一的缓存抽象,它允许开发者在不关心具体...

    JAVA编程之spring cache本机缓存应用

    1、SpringCache是Spring提供的一个缓存框架,在Spring3.1版本开始支持将缓存添加到现有的spring应用程序中,在4.1开始,缓存已支持JSR-107注释和更多自定义的选项 2、Spring Cache利用了AOP,实现了基于注解的缓存...

    Spring cache

    Spring Cache 是 Spring 3.1 版本引入的一项重要特性,它不是一种具体的缓存实现(如 EHCache 或 OSCache),而是一种缓存使用的抽象层。通过在现有的业务代码上添加特定的注解,可以轻松地为方法调用添加缓存支持。...

    Spring整合EhCache

    从 Spring 3.1 版本开始,Spring 框架正式引入了对 Cache 的支持,这种支持的方式和原理与 Spring 对事务管理的支持非常类似。通过这种方式,开发人员可以在不增加大量额外代码的情况下轻松地集成缓存技术,从而提高...

    Ehcache分布式缓存与其在spring中的使用

    ### Ehcache分布式缓存及其在Spring中的应用 #### 一、Ehcache概述与原理 Ehcache是一款高效且轻量级的纯Java缓存框架,由于其出色的性能和易于集成的特点,在Java开发中有着广泛的应用。作为Hibernate的默认缓存...

    Spring Boot 整合 Spring-cache:让你的网站速度飞起来.docx

    Spring Cache 是 Spring 框架的一个模块,自 3.1 版本起引入,提供了基于注解的缓存抽象,使得开发者能够方便地在应用中集成和管理缓存。 一、Spring Cache 介绍 Spring Cache 提供了一种统一的方式来管理不同类型...

    spring整合EhCache 简单例子

    Spring整合EhCache是一个常见的缓存管理方案,尤其在企业级Java应用中广泛使用。Spring框架提供了对多种缓存技术的支持,包括EhCache,这使得我们能够在Spring应用中轻松地实现数据缓存,提高系统性能。下面我们将...

    Spring来实现一个Cache简单的解决方案

    ### Spring实现Cache简单解决方案 #### 一、背景与概述 在现代软件开发中,缓存是一种常见的优化手段,用于提高应用程序的性能。Spring框架作为Java领域最流行的开发框架之一,为开发者提供了丰富的缓存管理机制。...

    点智数码--注释驱动的-Spring-cache-缓存的介绍.doc

    Spring Cache 是 Spring 框架从 3.1 版本开始引入的一种注解驱动的缓存抽象,它提供了一种简单而灵活的方式来在应用程序中实现缓存功能,无需依赖特定的缓存实现,如 EhCache 或 OSCache。Spring Cache 的核心特性...

    浅析SpringCache缓存1

    Spring Cache 是 Spring 框架从 3.1 版本开始引入的一个强大特性,它提供了一种透明化的缓存机制,允许开发者在不修改原有业务代码的基础上,轻松地为应用程序添加缓存功能。这个抽象层使得我们可以使用不同的缓存...

    Spring Cache的基本使用与实现原理详解

    Spring Cache 是Spring框架提供的一种缓存抽象,从Spring 3.1版本开始引入,目的是为了简化应用程序中的缓存管理,实现缓存透明化。通过在方法上添加特定注解,如@Cacheable、@CacheEvict等,可以轻松地启用缓存功能...

    详解Spring整合Ehcache管理缓存

    尽管 Ehcache 已经发布到了 3.1 版本,但本文将以 2.10.2 版本为基础进行讲解,因为 Spring 4.x 版本尚不直接支持 Ehcache 3.x。尽管可以通过 JCache API 进行间接整合,但这可能会增加复杂性。 ### 安装 Ehcache ...

    echache缓存注解说明文档

    要使用Spring Cache,我们需要在Spring配置中声明缓存管理器(如Ehcache、Redis等),并启用缓存注解处理。配置可以是基于XML的,也可以是基于Java配置。例如,使用Ehcache时,我们需要引入Ehcache库,并在配置中...

    spring3.1.4源码

    12. **缓存抽象**:Spring 3.1引入了缓存抽象,支持Ehcache、Hazelcast等缓存系统,相关接口位于`org.springframework.cache`包下。 通过阅读和分析Spring 3.1.4的源码,开发者可以深入了解Spring框架的设计理念和...

Global site tag (gtag.js) - Google Analytics