SpringBoot集成Redis和RedisTemplate

  

SpringBoot对redis非常友好,不需要像传统项目那样使用Jedis,只要添加spring-boot-starter-data-redis的maven依赖

<dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

接下来是添加redis和RedisTemplate的bean

package com.example.demo.config;

import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
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.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
import java.lang.reflect.Method;

/**
 * @Description:
 * @Author: zhaowang
 * @Date: 2018/10/8 15:42
 * @Version: 1.0
 */
@Configuration
@EnableCaching
public class RedisConfig extends CachingConfigurerSupport {

    @Bean
    public KeyGenerator keyGenerator() {
        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();
            }
        };
    }



    @Bean(name = "redisTemplate")
    public RedisTemplate<?, ?> 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;
    }

}

application.yml配置

spring:
    redis:
        #redis数据库索引(默认为0)
        database: 0
        #redis服务器地址
        host: ip
        #redis服务器连接端口
        port: 6379
        #redis连接密码
        password: 123456

接下来是使用SpringBoot注解实现redis的存取操作的实例

因为使用了Spring Data Jpa,所以这里给出repository层的代码,然后在service层调用该层方法。

package com.example.demo.dao;

import com.example.demo.entity.DatabusEntity;
import org.springframework.cache.annotation.CacheConfig;
import org.springframework.cache.annotation.CacheEvict;
import org.springframework.cache.annotation.CachePut;
import org.springframework.cache.annotation.Cacheable;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Modifying;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.query.Param;
import org.springframework.stereotype.Repository;
import org.springframework.transaction.annotation.Transactional;
import java.util.List;
import java.util.Map;

/**
 * @Description:
 * @Author: zhaowang
 * @Date: 2018/9/21 17:22
 * @Version: 1.0
 */
@Repository
@CacheConfig(cacheNames = "databusEntity")
public interface DatabusDao extends JpaRepository<DatabusEntity, Integer> {


    @Cacheable(key = "#p0", keyGenerator = "myKeyGenerator")
    public DatabusEntity findById(int id);

    @CachePut(key = "#p0.id") // 缓存插入的结果
    @Override
    @Transactional(value="transactionManager")
    public DatabusEntity save(DatabusEntity databusEntity);

    @CacheEvict(key="#p0", allEntries=true) // 删除对应的缓存
    @Transactional(value="transactionManager")
    public void deleteById(int id);


}

注解说明

@CacheConfig(cacheNames = "databusEntity")

cacheNames 指定该类所有方法操作的缓存名称,是全局的。也可以在方法上加,但缓存名称仅对该方法生效,是局部的。

 

 @Cacheable(key = "#p0", keyGenerator = "myKeyGenerator")

1.@Cacheable : 如果方法的第一个参数key(p0)存在于redis缓存中,则从缓存中获取value作为方法的返回结果,否则按照方法定义获取返回结果(一般是查询数据库)。

2.#p0表示第一个参数的key,即保存该方法操作的缓存的key(以HashMap的形式存储)到第一个参数,keyGenerator 定义Key生成的类。

 

@CachePut(key = "#p0.id")

p0.id表示将第一个参数的id作为key缓存到redis中,value为第一个参数值。

 

@CacheEvict(key="#p0", allEntries=true) 

1.p0表示删除key为第一个参数的redis缓存数据

2.allEntries=true表示删除缓存中的所有值(key和value)

allEntries=false表示仅删除key对应的value

 

测试类

package com.example.demo.controller;

import com.example.demo.dao.DatabusDao;
import com.example.demo.domain.DatabusTaskData;
import com.example.demo.entity.DatabusEntity;
import com.example.demo.service.DatabusService;
import com.example.demo.service.DatabusTaskDataService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.*;

/**
 * @Description: test
 * @Author: zhaowang
 * @Date: 2018/8/7 17:46
 * @version: 1.0
 */
@Slf4j
@RestController
@RequestMapping("demo/con")
public class TestController {

    @Autowired
    private DatabusTaskDataService databusTaskDataService;

    @Autowired
    private DatabusService databusService;

    @Autowired
    private DatabusDao databusDao;


     @RequestMapping("/cache")
    public void cache() {

        DatabusEntity databusEntity = databusDao.findById(3073);
        System.out.println("第一次执行:"+databusEntity.getFile_name());
        databusEntity.setFile_name("file1");
        databusDao.save(databusEntity);

        DatabusEntity databusEntity2 = databusDao.findById(3073);
        System.out.println("第二次执行:"+databusEntity2.getFile_name());

        databusDao.deleteById(3073);
        DatabusEntity databusEntity3 = databusDao.findById(3073);
        if(databusEntity3 != null)
            System.out.println("第三次执行:"+databusEntity3.getFile_name());
    }
}

访问url,对比结果可以清晰地知道redis的缓存put和evict的过程。

 

相关文章