详谈@Cacheable不起作用的原因:bean未序列化问题

目录
  • @Cacheable不起作用的原因:bean未序列化
    • 是返回的Blogger自定义实体类没有实现序列化接口
  • @Cacheable注解式缓存不起作用的情形
    • 使用注解式缓存的正确方式

@Cacheable不起作用的原因:bean未序列化

SpringMVC中将serviceImpl的方法返回值缓存在redis中,发现@Cacheable失效

是返回的Blogger自定义实体类没有实现序列化接口

无法存入到redis中。implements一下Serializable接口即可!

@Cacheable注解式缓存不起作用的情形

@Cacheable注解式缓存使用的要点:正确的注解式缓存配置,注解对象为spring管理的hean,调用者为另一个对象。有些情形下注解式缓存是不起作用的:同一个bean内部方法调用,子类调用父类中有缓存注解的方法等。后者不起作用是因为缓存切面必须走代理才有效,这时可以手动使用CacheManager来获得缓存效果。

使用注解式缓存的正确方式

<cache:annotation-driven cache-manager="springCacheManager" proxy-target-class="false"/>
<bean id="ehcacheManager" class="org.springframework.cache.ehcache.EhCacheManagerFactoryBean">
    <property name="configLocation" value="classpath:ehcache.xml"/>
    <property name="cacheManagerName" value="ehcache"/>
</bean>
<bean id="springCacheManager" class="org.springframework.cache.ehcache.EhCacheCacheManager">
    <property name="cacheManager" ref="ehcacheManager"/>
</bean>

要点:@Cacheable(value="必须使用ehcache.xml已经定义好的缓存名称,否则会抛异常")

@Component
public class CacheBean {
    @Cacheable(value="passwordRetryCache",key="#key")
    public String map(String key) {
        System.out.println("get value for key: "+key);
        return "value: "+key;
    }
    public String map2(String key) {
        return map(key);
    }
}
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = { "classpath:cache.xml" })
public class CacheTester {
    @Autowired CacheManager cacheManager;
    @Autowired CacheBean cacheBean;
    @Test public void cacheManager() {
        System.out.println(cacheManager);
    }
    @Test public void cacheAnnotation() {
        cacheBean.map("a");
        cacheBean.map("a");
        cacheBean.map("a");
        cacheBean.map("a");
        System.out.println(cacheManager.getCacheNames());
    }
}

输出:

get value for key: a
[authorizationCache, authenticationCache, shiro-activeSessionCache, passwordRetryCache]

稍微改造一下,让ehcache支持根据默认配置自动添加缓存空间,这里提供自定义的MyEhCacheCacheManager即可

<bean id="springCacheManager" class="com.itecheast.ite.domain.util.MyEhCacheCacheManager">
    <property name="cacheManager" ref="ehcacheManager"/>
</bean>

另一种改造方式,找不到已定义的缓存空间时不缓存,或者关闭全部缓存。把cacheManagers配置去掉就可以关闭圈闭缓存。

<bean id="springCacheManager" class="org.springframework.cache.support.CompositeCacheManager">
    <property name="cacheManagers">
        <list>
            <bean class="org.springframework.cache.ehcache.EhCacheCacheManager"></bean>
            <!-- <bean class="com.itecheast.ite.domain.util.MyEhCacheCacheManager"></bean> 这个会自动创建缓存空间 -->
        </list>
    </property>
    <property name="fallbackToNoOpCache" value="true"/>
</bean>

调用相同类或父类方法没有缓存效果:这时可以选择手动使用CacheManager。

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = { "classpath:cache.xml" })
public class CacheTester {
    @Test public void cacheAnnotation() {
        this.map("a");
        this.map("a");
    }
    @Cacheable(value="passwordRetryCache",key="#key")
    public String map(String key) {
        System.out.println("get value for key: "+key);
        return "value: "+key;
    }
}

或者再换一种方式:手动使用代理方式调用同类方法也是可以的

public class CacheBean {
    @Autowired ApplicationContext applicationContext;
    @Cacheable(value="passwordRetryCache",key="#key")
    public String map(String key) {  //方法不能为private,否则也没有缓存效果
        System.out.println("get value for key: "+key);
        return "value: "+key;
    }
    public String map2(String key) {
        CacheBean proxy = applicationContext.getBean(CacheBean.class);
        return proxy.map(key); //这里使用proxy调用map就可以缓存,而直接调用map则没有缓存
    }
}

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • springboot增加注解缓存@Cacheable的实现

    目录 springboot增加注解缓存@Cacheable 业务层使用 配置 @Cacheable注解的属性使用 cacheNames和value key keyGenerator keyGenerator condition unless(除非) sync springboot增加注解缓存@Cacheable 业务层使用 @Cacheable(value = "dictionary#1800", key = "#root.targetClass.simpleName +':

  • 解决springCache配置中踩的坑

    目录 springCache配置中踩的坑 先附上正确的配置 springCache配置及一些问题的解决 配置 @Cacheable参数 @CacheEvict 参数 @CachePut 参数 springCache配置中踩的坑 项目基于SpringBoot,使用了SpringCache. 早先在网上找了一份SpringCache的配置,后来由于需要使用到自定义序列化方法,注入一个自定义的序列化类.但是在后来发现自定义的序列化类始终没有调用,后来查看源码后终于发现了原因 先附上正确的配置 @Bea

  • Spring @Cacheable redis异常不影响正常业务方案

    背景 项目中,使用@Cacheable进行数据缓存.发现:当redis宕机之后,@Cacheable注解的方法并未进行缓存冲突,而是直接抛出异常.而这样的异常会导致服务不可用. 原因分析 我们是通过@EnableCaching进行缓存启用的,因此可以先看@EnableCaching的相关注释 通过@EnableCaching的类注释可发现,spring cache的核心配置接口为:org.springframework.cache.annotation.CachingConfigurer /**

  • 解决@Cacheable在同一个类中方法调用不起作用的问题

    @Cacheable在同一类中方法调用无效 上述图片中,同一个类中genLiveBullets()方法调用同类中的queryLiveByRoom()方法,这样即便标识了Cacheable标签,再次调用时也没有走缓存. cacheable不支持内部方法调用的方式,需要修改为把请求缓存的改成service方式,Aservice中的genLiveBullets()调用Bservice中的queryLiveByRoom() 见下图,已测试再次调用时为走的缓存. @Cacheable的使用总结 @Cach

  • 详谈@Cacheable不起作用的原因:bean未序列化问题

    目录 @Cacheable不起作用的原因:bean未序列化 是返回的Blogger自定义实体类没有实现序列化接口 @Cacheable注解式缓存不起作用的情形 使用注解式缓存的正确方式 @Cacheable不起作用的原因:bean未序列化 SpringMVC中将serviceImpl的方法返回值缓存在redis中,发现@Cacheable失效 是返回的Blogger自定义实体类没有实现序列化接口 无法存入到redis中.implements一下Serializable接口即可! @Cacheab

  • 详谈php ip2long 出现负数的原因及解决方法

    php提供了ip2long与long2ip方法对ip地址处理. 1.ip2long - 将一个IPV4的字符串互联网协议转换成数字格式 int ip2long ( string $ip_address ) 参数: ip_address 一个标准格式的地址. 返回值: 返回IP地址转换后的数字 或 FALSE 如果 ip_address 是无效的. 2.long2ip - 将数字格式转换成一个IPV4的字符串互联网协议 string long2ip ( string $proper_address

  • Angular中innerHTML标签的样式不起作用的原因解析

    1.背景 在最近angular的项目中,需要用到[innerHTML]标签来指定一个div的样式: //HTML部分 <div class="contents" [innerHTML]="contents"></div> //TS部分 contents = '<p>商品信息栏位<br><span style="color:red;">商品信息介绍</span></p&g

  • vant组件库之tag渐变色不起作用的原因及解决

    目录 tag渐变色不起作用的原因 方案一 在不影响原来功能的前提下修改源码 方案二 利用 /deep/ 修改组件的样式 vue渐变色背景样式 两种颜色渐变 三种颜色渐变 tag渐变色不起作用的原因 查看源码 从源码部分可以看出,如果传入 plain 属性则color 为字体颜色,反之为背景颜色, 用的是 backgroundColor 并不是 backgroundImage 所以背景并不能使用渐变色作为背景色 明白原理之后处理就好很多了 方案一 在不影响原来功能的前提下修改源码 将 var ke

  • @Transactional注解不起作用的原因分析及解决

    目录 Transactional失效场景介绍 第一种 第二种 第三种 @Transactional注解不起作用原理分析 第一种 不创建代理对象 不进行代理调用 第二种 第三种 Transactional失效场景介绍 第一种 Transactional注解标注方法修饰符为非public时,@Transactional注解将会不起作用.例如以下代码. 定义一个错误的@Transactional标注实现,修饰一个默认访问符的方法 /** * @author zhoujy * @date 2018年12

  • 关于vue状态过渡transition不起作用的原因解决

    总所周知,vue中的transition标签可以方便得进行动画过渡,使用的方法也很简单. <transition name="你要的名字"> 过渡的元素... </transition> 这里需要主要一点的是:过渡的元素只能是以下之一: 条件渲染 (使用 v-if) 条件展示 (使用 v-show) 动态组件 组件根节点 常用的过渡名称有fade等 你可以这样用 <transition name="fade"> 过渡的元素...

  • 详解常用的Spring Bean扩展接口

    前言 Spring是一款非常强大的框架,可以说是几乎所有的企业级Java项目使用了Spring,而Bean又是Spring框架的核心. Spring框架运用了非常多的设计模式,从整体上看,它的设计严格遵循了OCP----开闭原则,即: 1.保证对修改关闭,即外部无法修改Spring整个运作的流程 2.提供对扩展开放,即可以通过继承.实现Spring提供的众多抽象类与接口来改变类加载的行为 开卷有益,阅读Spring源码(无需每个类都看得很细,大体流程能梳理出来即可)对于个人水平的提升是帮助非常大

  • SpringBoot内部调用事务不起作用问题的解决方案

    在做业务开发时,遇到了一个事务不起作用的问题.大概流程是这样的,方法内部的定时任务调用了一个带事务的方法,失败后事务没有回滚.查阅资料后,问题得到解决,记录下来分享给大家. 场景 我在这里模拟一个场景,大概的调用方式就如下面的代码这样. @Override @Transactional(rollbackFor = RuntimeException.class) public void insertUser(User user) { userMapper.insertUser(user); thr

  • 深入了解Spring的Bean生命周期

    目录 源码下载 什么是 Spring Bean 的生命周期 Bean的生命周期 Spring角度查看bean的定义与注册 SpringBoot角度查看bean定义和注册 实例化,依赖注入,初始化 Bean的生命周期的扩展点 Bean级别 容器级别 常用接口 InstantiationAwareBeanPostProcessor BeanNameAware BeanFactoryAware ApplicationContextAware BeanPostProcessor Initializing

  • 详解Spring中Bean的加载的方法

    之前写过bean的解析,这篇来讲讲bean的加载,加载要比bean的解析复杂些,从之前的例子开始. Spring中加载一个bean的方式: TestBean bean = factory.getBean("testBean"); 来看看getBean(String name)方法源码, @Override public Object getBean(String name) throws BeansException { return doGetBean(name, null, nul

随机推荐