Spring Boot使用redis实现数据缓存

基于Spring Boot 1.5.2.RELEASE版本,一方面验证与Redis的集成方法,另外了解使用方法。

集成方法

  1. 配置依赖

    修改pom.xml,增加如下内容。

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
        </dependency>  
  2. 配置Redis

    修改application.yml,增加如下内容。

    spring:
        redis:
            host: localhost
            port: 6379
            pool:
                max-idle: 8
                min-idle: 0
                max-active: 8
                max-wait: -1
  3. 配置Redis缓存
    
    package net.jackieathome.cache;
    
    import java.lang.reflect.Method;
    
    import org.springframework.cache.CacheManager;
    import org.springframework.cache.annotation.CachingConfigurerSupport;
    import org.springframework.cache.annotation.EnableCaching;
    import org.springframework.cache.interceptor.KeyGenerator;
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    import org.springframework.data.redis.cache.RedisCacheManager;
    import org.springframework.data.redis.connection.RedisConnectionFactory;
    import org.springframework.data.redis.core.RedisTemplate;
    import org.springframework.data.redis.core.StringRedisTemplate;
    import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
    
    import com.fasterxml.jackson.annotation.JsonAutoDetect;
    import com.fasterxml.jackson.annotation.PropertyAccessor;
    import com.fasterxml.jackson.databind.ObjectMapper;
    
    @Configuration
    @EnableCaching // 启用缓存特性
    public class RedisConfig extends CachingConfigurerSupport {
        // 缓存数据时Key的生成器,可以依据业务和技术场景自行定制
    //  @Bean
    //  public KeyGenerator customizedKeyGenerator() {
    //      return new KeyGenerator() {
    //          @Override
    //          public Object generate(Object target, Method method, Object... params) {
    //              StringBuilder sb = new StringBuilder();
    //              sb.append(target.getClass().getName());
    //              sb.append(method.getName());
    //              for (Object obj : params) {
    //                  sb.append(obj.toString());
    //              }
    //              return sb.toString();
    //          }
    //      };
    //
    //  }
        // 定制缓存管理器的属性,默认提供的CacheManager对象可能不能满足需要
        // 因此建议依赖业务和技术上的需求,自行做一些扩展和定制
        @Bean
        public CacheManager cacheManager(@SuppressWarnings("rawtypes") RedisTemplate redisTemplate) {
            RedisCacheManager redisCacheManager = new RedisCacheManager(redisTemplate);
            redisCacheManager.setDefaultExpiration(300);
            return redisCacheManager;
        }
    
        @Bean
        public RedisTemplate<String, String> redisTemplate(RedisConnectionFactory factory) {
            StringRedisTemplate template = new StringRedisTemplate(factory);
            Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
            ObjectMapper om = new ObjectMapper();
            om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
            om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
            jackson2JsonRedisSerializer.setObjectMapper(om);
            template.setValueSerializer(jackson2JsonRedisSerializer);
            template.afterPropertiesSet();
            return template;
        }
    }

验证集成后的效果

考虑到未来参与的项目基于MyBatis实现数据库访问,而利用缓存,可有效改善Web页面的交互体验,因此设计了如下两个验证方案。

方案一

在访问数据库的数据对象上增加缓存注解,定义缓存策略。从测试效果看,缓存有效。

  1. 页面控制器

    
    package net.jackieathome.controller;
    
    import java.util.List;
    
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestMethod;
    import org.springframework.web.bind.annotation.RestController;
    
    import net.jackieathome.bean.User;
    import net.jackieathome.dao.UserDao;
    import net.jackieathome.db.mapper.UserMapper;
    
    @RestController
    public class UserController {
    
        @Autowired
        private UserDao userDao;
    
        @RequestMapping(method = RequestMethod.GET, value = "/user/id/{id}")
        public User findUserById(@PathVariable("id") String id) {
            return userDao.findUserById(id);
        }
    
        @RequestMapping(method = RequestMethod.GET, value = "/user/create")
        public User createUser() {
            long time = System.currentTimeMillis() / 1000;
    
            String id = "id" + time;
            User user = new User();
            user.setId(id);
            userDao.createUser(user);
    
            return userDao.findUserById(id);
        }
    }
  2. Mapper定义
    
    package net.jackieathome.db.mapper;
    
    import java.util.List;
    
    import org.apache.ibatis.annotations.Mapper;
    import org.apache.ibatis.annotations.Param;
    import org.springframework.cache.annotation.CacheConfig;
    import org.springframework.cache.annotation.CachePut;
    import org.springframework.cache.annotation.Cacheable;
    
    import net.jackieathome.bean.User;
    
    @Mapper
    public interface UserMapper {
    
        void createUser(User user);
    
        User findUserById(@Param("id") String id);
    }
  3. 数据访问对象
    
    package net.jackieathome.dao;
    
    import java.util.ArrayList;
    import java.util.List;
    
    import org.apache.ibatis.annotations.Param;
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.cache.annotation.CacheConfig;
    import org.springframework.cache.annotation.CachePut;
    import org.springframework.cache.annotation.Cacheable;
    import org.springframework.stereotype.Component;
    import org.springframework.transaction.annotation.Transactional;
    
    import net.jackieathome.bean.User;
    import net.jackieathome.db.mapper.UserMapper;
    
    @Component
    @CacheConfig(cacheNames = "users")
    @Transactional
    public class UserDao {
        private static final Logger LOG = LoggerFactory.getLogger(UserDao.class);
        @Autowired
        private UserMapper userMapper;
    
        @CachePut(key = "#p0.id")
        public void createUser(User user) {
            userMapper.createUser(user);
            LOG.debug("create user=" + user);
        }
    
        @Cacheable(key = "#p0")
        public User findUserById(@Param("id") String id) {
            LOG.debug("find user=" + id);
            return userMapper.findUserById(id);
        }
    }

方案二

直接在Mapper定义上增加缓存注解,控制缓存策略。从测试效果看,缓存有效,相比于方案一,测试代码更加简洁一些。

  1. 页面控制器

    
    package net.jackieathome.controller;
    
    import java.util.List;
    
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RequestMethod;
    import org.springframework.web.bind.annotation.RestController;
    
    import net.jackieathome.bean.User;
    import net.jackieathome.dao.UserDao;
    import net.jackieathome.db.mapper.UserMapper;
    
    @RestController
    public class UserController {
    
        @Autowired
        private UserMapper userMapper;
    
        @RequestMapping(method = RequestMethod.GET, value = "/user/id/{id}")
        public User findUserById(@PathVariable("id") String id) {
            return userMapper.findUserById(id);
        }
    
        @RequestMapping(method = RequestMethod.GET, value = "/user/create")
        public User createUser() {
            long time = System.currentTimeMillis() / 1000;
    
            String id = "id" + time;
            User user = new User();
            user.setId(id);
            userMapper.createUser(user);
    
            return userMapper.findUserById(id);
        }
    }
  2. Mapper定义
    package net.jackieathome.db.mapper;
    
    import java.util.List;
    
    import org.apache.ibatis.annotations.Mapper;
    import org.apache.ibatis.annotations.Param;
    import org.springframework.cache.annotation.CacheConfig;
    import org.springframework.cache.annotation.CachePut;
    import org.springframework.cache.annotation.Cacheable;
    
    import net.jackieathome.bean.User;
    
    @CacheConfig(cacheNames = "users")
    @Mapper
    public interface UserMapper {
    
        @CachePut(key = "#p0.id")
        void createUser(User user);
    
        @Cacheable(key = "#p0")
        User findUserById(@Param("id") String id);
    }

总结

上述两个测试方案并没有优劣之分,仅是为了验证缓存的使用方法,体现了不同的控制粒度,在实际的项目开发过程中,需要依据实际情况做不同的决断。

缓存相关的注解:

  1. CacheConfig
  2. Cacheable
  3. CachePut
  4. CacheEvict

参考资料

使用redis缓存

MyBatis的缓存特性

其它

  1. 运行redis docker镜像的命令

    sudo docker run --rm -d -p 6379:6379 redis

查看原文http://www.jackieathome.net/archives/485.html

时间: 2024-10-24 18:54:27

Spring Boot使用redis实现数据缓存的相关文章

Spring Boot 中集成 Redis 作为数据缓存

只添加注解:@Cacheable,不配置key时,redis 中默认存的 key 是:users::SimpleKey [](1.redis-cli 中,通过命令:keys * 查看:2.key:缓存对象存储在Map集合中的key值,非必需,缺省按照函数的所有参数组合作为key值,若自己配置需使用SpEL表达式,比如:@Cacheable(key = "#p0"):使用函数第一个参数作为缓存的key值,更多关于SpEL表达式的详细内容可参考官方文档). 相关文章 网址 SpringBo

spring boot 使用 redis 缓存

spring boot redis 使用 1 Redis:Redis 是完全开源免费的,遵守BSD协议,是一个高性能的key-value数据库.Redis 与其他 key - value 缓存产品有以下三个特点:Redis支持数据的持久化,可以将内存中的数据保存在磁盘中,重启的时候可以再次加载进行使用.Redis不仅仅支持简单的key-value类型的数据,同时还提供list,set,zset,hash等数据结构的存储.Redis支持数据的备份,即master-slave模式的数据备份. spr

Spring整合Redis做数据缓存(Windows环境)

当我们一个项目的数据量很大的时候,就需要做一些缓存机制来减轻数据库的压力,提升应用程序的性能,对于java项目来说,最常用的缓存组件有Redis.Ehcache和Memcached. Ehcache是用java开发的缓存组件,和java结合良好,直接在jvm虚拟机中运行,不需要额外安装什么东西,效率也很高:但是由于和java结合的太紧密了,导致缓存共享麻烦,分布式集群应用不方便,所以比较适合单个部署的应用. Redis需要额外单独安装,是通过socket访问到缓存服务,效率比Ehcache低,但

Spring Boot自定义Redis缓存配置,保存value格式JSON字符串

Spring Boot自定义Redis缓存,保存格式JSON字符串 部分内容转自 https://blog.csdn.net/caojidasabi/article/details/83059642 package springboot01cache.config; import com.fasterxml.jackson.annotation.JsonAutoDetect; import com.fasterxml.jackson.annotation.PropertyAccessor; im

SpringBoot(三) :Spring boot 中 Redis 的使用

前言: 这一篇讲的是Spring Boot中Redis的运用,之前没有在项目中用过Redis,所以没有太大的感觉,以后可能需要回头再来仔细看看. 原文出处: 纯洁的微笑 SpringBoot对常用的数据库支持外,对NoSQL 数据库也进行了封装自动化. redis介绍 Redis是目前业界使用最广泛的内存数据存储.相比memcached,Redis支持更丰富的数据结构,例如hashes, lists, sets等,同时支持数据持久化.除此之外,Redis还提供一些类数据库的特性,比如事务,HA,

Spring boot、Redis、ActiveMQ、Nginx、Mycat、Netty、Jvm大型分布式项目实战视频教程

15套java架构师.集群.高可用.高可扩展.高性能.高并发.性能优化.Spring boot.Redis.ActiveMQ.Nginx.Mycat.Netty.Jvm大型分布式项目实战视频教程 视频课程内容包含: 高级Java架构师包含:Spring boot.Spring  cloud.Dubbo.Redis.ActiveMQ.Nginx.Mycat.Spring.MongoDB.ZeroMQ.Git.Nosql.Jvm.Mecached.Netty.Nio.Mina.性能调优.高并发.to

【redis】spring boot利用redis的Keyspace Notifications实现消息通知

前言 需求:当redis中的某个key失效的时候,把失效时的value写入数据库. github: https://github.com/vergilyn/RedisSamples 1.修改redis.conf 安装的redis服务默认是: notify-keyspace-events "",修改成 notify-keyspace-events Ex; 位置:redis安装目下的redis.windows-service.conf 或 redis.windows.conf.(具体看re

Spring Boot学习笔记——Spring Boot与Redis的集成

一.添加Redis缓存 1.添加Redis起步依赖 在pom.xml中添加Spring Boot支持Redis的依赖配置,具体如下: <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-redis</artifactId> <version>1.4.7.RELEASE</version> </

Spring Boot 和 Redis 常用操作

1    第4-2课:Spring Boot 和 Redis 常用操作 Redis 是目前使用最广泛的缓存中间件,相比 Memcached,Redis 支持更多的数据结构和更丰富的数据操作,另外 Redis 有着丰富的集群方案和使用场景,这一课我们一起学习 Redis 的常用操作. 1.1    Redis 介绍 Redis 是一个速度非常快的非关系数据库(Non-Relational Database),它可以存储键(Key)与 5 种不同类型的值(Value)之间的映射(Mapping),可