文档章节

SpringBoot集成Redis来实现缓存技术方案

FEINIK
 FEINIK
发布于 2017/06/25 23:19
字数 1183
阅读 6096
收藏 422

概述

在我们的日常项目开发过程中缓存是无处不在的,因为它可以极大的提高系统的访问速度,关于缓存的框架也种类繁多,今天主要介绍的是使用现在非常流行的NoSQL数据库(Redis)来实现我们的缓存需求。

Redis简介

Redis 是一个开源(BSD许可)的,内存中的数据结构存储系统,它可以用作数据库、缓存和消息中间件,Redis 的优势包括它的速度、支持丰富的数据类型、操作原子性,以及它的通用性。

案例整合

本案例是在之前一篇SpringBoot + Mybatis + RESTful的基础上来集成Redis的,所以大家如有什么不明白的地方可以前往https://my.oschina.net/feinik/blog/879266,由于篇幅原因这里不一一贴出所有的代码,具体完整案例代码可以看这里:https://github.com/AIFEINIK/SpringBoot-Learn/tree/master/spring-boot-redis2,关于Redis如何安装可自行google。

1、在Maven pom.xml文件中加入Redis包

<!--redis-->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-redis</artifactId>
    <version>${boot.version}</version>
</dependency>

2、SpringBoot配置文件中配置Redis连接(YAML方式配置)

spring:
    application:
        name: spring-boot-redis
    redis:
        host: 192.168.145.132
        port: 6379
        timeout: 20000
        cluster:
            nodes: 192.168.211.134:7000,192.168.211.134:7001,192.168.211.134:7002
            maxRedirects: 6
        pool:
            max-active: 8
            min-idle: 0
            max-idle: 8
            max-wait: -1

解释:本配置采用Redis一主三从的的配置方式来提高缓存的吞吐量

3、Redis配置类

@Configuration
public class RedisConfig {

   @Bean
   public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory connectionFactory) {
      RedisTemplate<Object, Object> template = new RedisTemplate<>();
      template.setConnectionFactory(connectionFactory);

      //使用Jackson2JsonRedisSerializer来序列化和反序列化redis的value值
      Jackson2JsonRedisSerializer serializer = new Jackson2JsonRedisSerializer(Object.class);

      ObjectMapper mapper = new ObjectMapper();
      mapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
      mapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
      serializer.setObjectMapper(mapper);

      template.setValueSerializer(serializer);
      //使用StringRedisSerializer来序列化和反序列化redis的key值
      template.setKeySerializer(new StringRedisSerializer());
      template.afterPropertiesSet();
      return template;
   }
}

解释:SpringBoot提供了对Redis的自动配置功能,在RedisAutoConfiguration中默认为我们配置了JedisConnectionFactory(客户端连接)、RedisTemplate以及StringRedisTemplate(数据操作模板),其中StringRedisTemplate模板只针对键值对都是字符型的数据进行操作,本示例采用RedisTemplate作为数据操作模板,该模板默认采用JdkSerializationRedisSerializer的二进制数据序列化方式,为了方便演示本示例采用Jackson2JsonRedisSerializer来序列化和反序列化redis的value值,使用StringRedisSerializer来序列化和反序列化redis的key值。

4、Service层应用缓存(注解方式)

@Service
public class PersonService {

    @Autowired
    private PersonRepo personRepo;

   /**
     * @Cacheable 应用到读取数据的方法上,先从缓存中读取,如果没有再从DB获取数据,然后把数据添加到缓存中
    * unless 表示条件表达式成立的话不放入缓存
     * @param username
     * @return
     */
    @Cacheable(value = "user", key = "#root.targetClass + #username", unless = "#result eq null")
    public Person getPersonByName(String username) {
        Person person = personRepo.getPersonByName(username);
        return person;
    }

   /**
    * @CachePut 应用到写数据的方法上,如新增/修改方法,调用方法时会自动把相应的数据放入缓存
     * @param person
     * @return
     */
    @CachePut(value = "user", key = "#root.targetClass + #result.username", unless = "#person eq null")
    public Person savePerson(Person person) {
        return personRepo.savePerson(person);
    }

   /**
    * @CacheEvict 应用到删除数据的方法上,调用方法时会从缓存中删除对应key的数据
     * @param username
     * @return
     */
    @CacheEvict(value = "user", key = "#root.targetClass + #username", condition = "#result eq true")
    public boolean removePersonByName(String username) {
        return personRepo.removePersonByName(username) > 0;
    }

    public boolean isExistPersonName(Person person) {
        return personRepo.existPersonName(person) > 0;
    }
}

解释:

1、这里的缓存key为简单的字符串组合,也可根据具体需要实现自定义的Key生成器,然后在注解中使用keyGenerator来引用。

2、Spring Cache提供了一些供我们使用的SpEL上下文数据,通过#来引用,具体可查看Spring官网:http://docs.spring.io/spring/docs/current/spring-framework-reference/htmlsingle/#cache-spel-context。

5、数据访问资源类

@Component
@Path("personMgr")
public class PersonMgrResource {

    @Autowired
    private PersonService personService;

    @GET
    @Path("getPersonByName")
    @Produces(MediaType.APPLICATION_JSON)
    public JsonResp getPersonByName(@QueryParam("username") String username) {
        Person person = personService.getPersonByName(username);
        return JsonResp.success(person);
    }

    @POST
    @Path("removePersonByName")
    @Produces(MediaType.APPLICATION_JSON)
    public JsonResp removePersonByName(@QueryParam("username") String username) {
        if (personService.removePersonByName(username)) {
            return JsonResp.success();
        }
        return JsonResp.fail("系统错误!");
    }

    @POST
    @Path("savePerson")
    @Produces(MediaType.APPLICATION_JSON)
    public JsonResp savePerson(Person person) {
        if (personService.isExistPersonName(person)) {
            return JsonResp.fail("用户名已存在!");
        }
        if (personService.savePerson(person).getId() > 0) {
            return JsonResp.success();
        }
        return JsonResp.fail("系统错误!");
    }
}

6、通过postman工具来测试缓存是否生效

第一次访问查找用户:

第一次通过用户名称来查找用户可以看到是从库中查询的数据,我们可以通过RedisClient工具来查看数据已放入了缓存

第二次查找用户:发现服务端并未打印任何数据库查询日志,可以知道第二次查询是从缓存中查询得到的数据。

总结

本文介绍如何通过SpringBoot来一步步集成Redis缓存,关于Redis的使用它不仅可以用作缓存,还可以用来构建队列系统,Pub/Sub实时消息系统,分布式系统的的计数器应用,关于Redis更多的介绍,请前往查阅官方文档。

 

© 著作权归作者所有

FEINIK
粉丝 228
博文 59
码字总数 56632
作品 0
广州
高级程序员
私信 提问
加载中

评论(30)

吴峻申
吴峻申
如果我要用redis做缓存的项目不是一个数据库项目,而且我想在controller这一层读取或放入数据到redis缓存,那么该如何实现?老实讲,我个人觉得你这篇文章价值并不大
FEINIK
FEINIK 博主

引用来自“吕兵阳”的评论

另外,spring封装的redis模板是有问题的在操作字符串的时候,会在最前面加上一个二进制类型,除非你指定序列化类型。

引用来自“FEINIK”的评论

SpringBoot为我们自动配置了RedisTemplate,该模板使用JdkSerializationRedisSerializer(二进制方式)作为默认的序列化方式,如果我们使用RedisClient查看的话会看到类似乱码的东西,你可以当做是编码的问题,会影响我们直观的查看,所以为了更加直观的查看到加入的缓存内容,这里可以指定其他的序列化方式,如Jackson2JsonRedisSerializer

引用来自“StackShao”的评论

自定义序列化方式为FastJson2JsonRedisSerializer,在存储HttpClient工具类中的CookieStore时,提示没有无参构造方法;又使用StringRedisTemplate,将cookieStore对象转换为字符串进行存储,发现从缓存中取出转换为BasicCookieStore类型对象后使用,发现cookie信息无效,最后还是使用默认的JdkSerializationRedisSerializer进行序列化,只是存储的key,value类似乱码,不过有效,不知道有没有更好的优化方法?
不妨将你的异常信息发出来看看
寒武没有纪
寒武没有纪

引用来自“吕兵阳”的评论

另外,spring封装的redis模板是有问题的在操作字符串的时候,会在最前面加上一个二进制类型,除非你指定序列化类型。

引用来自“FEINIK”的评论

SpringBoot为我们自动配置了RedisTemplate,该模板使用JdkSerializationRedisSerializer(二进制方式)作为默认的序列化方式,如果我们使用RedisClient查看的话会看到类似乱码的东西,你可以当做是编码的问题,会影响我们直观的查看,所以为了更加直观的查看到加入的缓存内容,这里可以指定其他的序列化方式,如Jackson2JsonRedisSerializer
自定义序列化方式为FastJson2JsonRedisSerializer,在存储HttpClient工具类中的CookieStore时,提示没有无参构造方法;又使用StringRedisTemplate,将cookieStore对象转换为字符串进行存储,发现从缓存中取出转换为BasicCookieStore类型对象后使用,发现cookie信息无效,最后还是使用默认的JdkSerializationRedisSerializer进行序列化,只是存储的key,value类似乱码,不过有效,不知道有没有更好的优化方法?
qwfys
qwfys
~~
j
jackygurui
为什么不用Redisson的现成SpringCache方案,反而自己去造轮子呢?
zcqshine
zcqshine

引用来自“吕兵阳”的评论

另外,spring封装的redis模板是有问题的在操作字符串的时候,会在最前面加上一个二进制类型,除非你指定序列化类型。

引用来自“FEINIK”的评论

SpringBoot为我们自动配置了RedisTemplate,该模板使用JdkSerializationRedisSerializer(二进制方式)作为默认的序列化方式,如果我们使用RedisClient查看的话会看到类似乱码的东西,你可以当做是编码的问题,会影响我们直观的查看,所以为了更加直观的查看到加入的缓存内容,这里可以指定其他的序列化方式,如Jackson2JsonRedisSerializer
原来如此, 难怪我每次用redis命令查看的时候 key 里面都有二进制编码
jarchan
jarchan
用redis做spring的cache层有个坑,就是不会自动flush,flush的动作得程序自己做,同时缺少flush策略。
如果集成ehcache之类的专用缓存工具,则不需要,因为这些功能ehcache会替你处理。
总而言之就是不推荐使用redis作为spring的cache层,开发者要自己造的轮子比较多。
每周精粹
每周精粹

引用来自“FEINIK”的评论

引用来自“开源中国首席董事长”的评论

@Cacheable 中的 value 和 key 有什么关系/?
我们知道 Redis 是 key value 的形式 , 那 @Cacheable 配置的 value 是做什么的
@Cacheable 注解中的value是指定缓存的名称,使用过Ehcache的话,都知道我们一般会根据不同的业务点来配置不同名称的缓存,然后使用CacheManager来管理着多个不同名称的缓存,key就是对应缓存名称下的键,可以这么理解value指定的缓存名称可以当做是java中的Map集合,而key就相当于Map中的键,缓存的对象就相当于Map中的value值😄

……redis 中有什么作用?
FEINIK
FEINIK 博主

引用来自“开源中国首席董事长”的评论

@Cacheable 中的 value 和 key 有什么关系/?
我们知道 Redis 是 key value 的形式 , 那 @Cacheable 配置的 value 是做什么的
@Cacheable 注解中的value是指定缓存的名称,使用过Ehcache的话,都知道我们一般会根据不同的业务点来配置不同名称的缓存,然后使用CacheManager来管理着多个不同名称的缓存,key就是对应缓存名称下的键,可以这么理解value指定的缓存名称可以当做是java中的Map集合,而key就相当于Map中的键,缓存的对象就相当于Map中的value值😄
SpringBootBucket 1.0.0 发布,SprintBoot 全家桶

Spring Boot 现在已经成为Java 开发领域的一颗璀璨明珠,它本身是包容万象的,可以跟各种技术集成。 本项目对目前Web开发中常用的各个技术,通过和SpringBoot的集成,并且对各种技术通过“一...

一刀
2018/03/05
8.7K
17
Spring Boot 全家桶 - SpringBootBucket

Spring Boot 现在已经成为Java 开发领域的一颗璀璨明珠,它本身是包容万象的,可以跟各种技术集成。 本项目对目前Web开发中常用的各个技术,通过和SpringBoot的集成,并且对各种技术通过“一...

一刀
2018/03/05
10.9K
8
SpringBootBucket 2.0.4 发布,代号“傲娇的小二晶”

SpringBootBucket 自从1.0.0版本发布后就有好多人喜欢,目前码云上面star数量接近1.2k。上个月还收到了红薯签名的1000 star奖杯,这个我自己也觉得很惊讶。 由于SpringBoot 1.x官方将终止维护...

一刀
2018/09/16
2.7K
4
SpringBoot注解方式整合Redis--最简单的实例

参考网页 springboot中注解实现集成redis https://blog.csdn.net/a67474506/article/details/52608855 一些SpringBoot相关知识的讲解 SpringBoot开箱即用的体现--本质上是SpringBoot提前做好...

karma123
2018/07/04
1K
0
恒宇少年/spring-boot-chapter

简书整套文档以及源码解析 专题 专题名称 专题描述 001 Spring Boot 核心技术 讲解SpringBoot一些企业级层面的核心组件 002 Spring Cloud 核心技术 对Spring Cloud核心技术全面讲解 003 Quer...

恒宇少年
2018/04/19
0
0

没有更多内容

加载失败,请刷新页面

加载更多

64.监控平台介绍 安装zabbix 忘记admin密码

19.1 Linux监控平台介绍 19.2 zabbix监控介绍 19.3/19.4/19.6 安装zabbix 19.5 忘记Admin密码如何做 19.1 Linux监控平台介绍: 常见开源监控软件 ~1.cacti、nagios、zabbix、smokeping、ope...

oschina130111
今天
11
0
当餐饮遇上大数据,嗯真香!

之前去开了一场会,主题是「餐饮领袖新零售峰会」。认真听完了餐饮前辈和新秀们的分享,觉得获益匪浅,把脑子里的核心纪要整理了一下,今天和大家做一个简单的分享,欢迎感兴趣的小伙伴一起交...

数澜科技
今天
7
0
DNS-over-HTTPS 的下一代是 DNS ON BLOCKCHAIN

本文作者:PETER LAI ,是 Diode 的区块链工程师。在进入软件开发领域之前,他主要是在做工商管理相关工作。Peter Lai 也是一位活跃的开源贡献者。目前,他正在与 Diode 团队一起开发基于区块...

红薯
今天
8
0
CC攻击带来的危害我们该如何防御?

随着网络的发展带给我们很多的便利,但是同时也带给我们一些网站安全问题,网络攻击就是常见的网站安全问题。其中作为站长最常见的就是CC攻击,CC攻击是网络攻击方式的一种,是一种比较常见的...

云漫网络Ruan
今天
11
0
实验分析性专业硕士提纲撰写要点

为什么您需要研究论文的提纲? 首先当您进行研究时,您需要聚集许多信息和想法,研究论文提纲可以较好地组织你的想法, 了解您研究资料的流畅度和程度。确保你写作时不会错过任何重要资料以此...

论文辅导员
今天
8
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部