简单的使用Redis写入和查询数据

本文介绍了如何使用Spring Boot的ShardingRedisUtil将字典数据分组写入Redis,以及如何根据字典类型查询数据。通过RedisTemplate操作哈希映射和列表,实现高效的数据管理和检索。
摘要由CSDN通过智能技术生成

我们以将字典数据写入Redis中,并查询数据为例

1、查询字典数据并写入Redis中

    @Resource
    private TestMapper testMapper;

    @Resource
    ShardingRedisUtil shardingRedisUtil;

    @Override
    public void testSetRedis() {
        // 获取所有字典信息
        List<TbSysDictInfo> list = testMapper.selectAll();
        // 根据字典类型分组
        Map<String, List<TbSysDictInfo>> distMap = list.stream()
                .collect(Collectors.groupingBy(TbSysDictInfo::getTypeCode));
        // 将数据写入redis
        shardingRedisUtil.hMSet(Dictconstants.DICT_RESULT, distMap);
    }

Dictconstants方法

public class Dictconstants {

    /**
     * 字典结果属性
     */
    public static final String DICT_RESULT = "dict_result_";

}

2、根据字典类型从Redis中查询数据

    @Override
    public List<TbSysDictInfo> testGetRedis(String type) {
        // 根据字典类型获取列表
        Map<String, List<TbSysDictInfo>> distMap = shardingRedisUtil.hMGet(new TypeReference<List<TbSysDictInfo>>() {
        }, Dictconstants.DICT_RESULT, type);
        if (distMap.get(type)!=null){
            return distMap.get(type);
        }else {
            return new ArrayList<>();
        }
    }

3、RedisUtil工具类 ShardingRedisUtil 

这个工具类功能很强大,大家可以自行挖掘


import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.exception.ExceptionUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.data.redis.core.Cursor;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.ScanOptions;
import org.springframework.stereotype.Component;
import org.springframework.util.CollectionUtils;

import javax.annotation.Resource;
import java.util.*;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;

@Component
public class ShardingRedisUtil {

    Logger logger = LoggerFactory.getLogger(this.getClass());

    @Resource
    private RedisTemplate<String, Object> redisTemplate;

    private static final int SHARDING_COUNT = 100; //分片数
    private static final int SCAN_COUNT = 1000; //扫描分批数
    private ObjectMapper objectMapper = new ObjectMapper();

    public void hMSet(String key, Map<String, ? extends Object> map) {
        hMSet(key, map, SHARDING_COUNT);
    }

    public <T> Map<String, T> hMGet(TypeReference<T> typeReference, String key, String... field) {
        return hMGet(typeReference, SHARDING_COUNT, key, field);
    }

    public <T> Map<String, T> hMGet(TypeReference<T> typeReference, String key, List<String> fields) {
        if (CollectionUtils.isEmpty(fields)) {
            return null;
        }
        return hMGet(typeReference, key, fields.toArray(new String[1]));
    }

    public long hDel(String key, String... field) {
        return hDel(SHARDING_COUNT, key, field);
    }

    public void hMSet(String key, Map<String, ? extends Object> map, int shardingCount) {
        if (!CollectionUtils.isEmpty(map)) {
            Map<String, Map<String, Object>> tmpMap = new HashMap<>();
            for (Map.Entry<String, ? extends Object> entry : map.entrySet()) {
                String tmpKey = entry.getKey();
                Object tmpValue = entry.getValue();
                int num = Math.abs(tmpKey.hashCode() % shardingCount);
                String hashKey = String.format("%s$%s", key, num);
                tmpMap.computeIfAbsent(hashKey, k -> new HashMap<>()).put(tmpKey, tmpValue);
            }
            for (Map.Entry<String, Map<String, Object>> entry : tmpMap.entrySet()) {
                redisTemplate.opsForHash().putAll(entry.getKey(), entry.getValue());
            }
        }
    }

    public void hMSetToOneGroup(String key, Map<String, Object> map) {
        if (!CollectionUtils.isEmpty(map)) {
            redisTemplate.opsForHash().putAll(key, map);
        }
    }

    public <T> Map<String, T> hMGetfromOneGroup(TypeReference<T> typeReference, String key, String... field) {
        Map<String, T> resultMap = new HashMap<>();
        if (field != null) {
            List<Object> objects = redisTemplate.opsForHash().multiGet(key, Arrays.asList(field));
            List<String> fieldList = new ArrayList<>();
            Collections.addAll(fieldList, field);
            for (int i = 0; i < fieldList.size(); i++) {
                Object o = objects.get(i);
                resultMap.put(fieldList.get(i), o != null ? objectMapper.convertValue(o, typeReference) : null);
            }
        }
        return resultMap;
    }

    public <T> Map<String, T> hMGet(TypeReference<T> typeReference, int shardingCount, String key, String... field) {
        Map<String, T> resultMap = new HashMap<>();
        if (field != null) {
            for (Map.Entry<String, List<String>> entry : separateToMap(shardingCount, key, field).entrySet()) {
                List<String> fieldList = entry.getValue();
                List<Object> objects = redisTemplate.opsForHash().multiGet(entry.getKey(), new ArrayList<>(fieldList));
                for (int i = 0; i < fieldList.size(); i++) {
                    Object o = objects.get(i);
                    resultMap.put(fieldList.get(i), o != null ? objectMapper.convertValue(o, typeReference) : null);
                }
            }
        }
        return resultMap;
    }

    /**
     * 扫描key前缀,获取key的所有redis缓存
     *
     * @param keyPattern
     * @param <T>
     * @return
     */
    public <T> Map<String, T> hMGet(TypeReference<T> typeReference, String keyPattern) {
        return hMGet(typeReference, SHARDING_COUNT, keyPattern);
    }

    /**
     * 扫描key前缀,获取key的所有redis缓存
     *
     * @param keyPattern
     * @param <T>
     * @return
     */
    public <T> Map<String, T> hMGet(TypeReference<T> typeReference, int shardingCount, String keyPattern) {
        Map<String, T> resultMap = new HashMap<>();
        if (StringUtils.isNoneBlank(keyPattern)) {
            try {
                for (int num = 0; num < shardingCount; num++) {
                    String hashKey = String.format("%s$%s", keyPattern, num);
//                    Cursor<Map.Entry<Object, Object>> cursor = redisTemplate.opsForHash().scan(hashKey, ScanOptions.scanOptions().count(SCAN_COUNT).build());
//                    while (cursor.hasNext()) {
//                        Map.Entry<Object, Object> next = cursor.next();
//                        Object field = next.getKey();
//                        Object value = next.getValue();
//                        if (field != null && value != null) {
//                            resultMap.put(field.toString(), objectMapper.convertValue(value, typeReference));
//                        }
//                    }
//                    cursor.close();
                    resultMap.putAll(hMGetTargetKey(typeReference,hashKey));
                }
            } catch (Exception e) {
                logger.error("扫描redis获取hashKey所有缓存异常", e);
            }
        }
        return resultMap;
    }

    /**
     * 扫描key前缀,获取key的所有redis缓存
     *
     * @param key
     * @param <T>
     * @return
     */
    public <T> Map<String, T> hMGetTargetKey(TypeReference<T> typeReference, String key) {
        Map<String, T> resultMap = new HashMap<>();
        if (StringUtils.isNoneBlank(key)) {
            try {
                Cursor<Map.Entry<Object, Object>> cursor = redisTemplate.opsForHash().scan(key, ScanOptions.scanOptions().count(SCAN_COUNT).build());
                while (cursor.hasNext()) {
                    Map.Entry<Object, Object> next = cursor.next();
                    Object field = next.getKey();
                    Object value = next.getValue();
                    if (field != null && value != null) {
                        resultMap.put(field.toString(), objectMapper.convertValue(value, typeReference));
                    }
                }
                cursor.close();
            } catch (Exception e) {
                logger.error("扫描redis获取hashKey所有缓存异常", e);
            }
        }
        return resultMap;
    }


    public Long hDel(int shardingCount, String key, String... field) {
        long count = 0;
        if (field != null) {
            for (Map.Entry<String, List<String>> entry : separateToMap(shardingCount, key, field).entrySet()) {
                count += redisTemplate.opsForHash().delete(entry.getKey(), entry.getValue().toArray());
            }
        }
        return count;
    }

    private Map<String, List<String>> separateToMap(int shardingCount, String key, String[] field) {
        Map<String, List<String>> tmpMap = new HashMap<>();
        for (String s : field) {
            if (StringUtils.isNotBlank(s)) {
                int num = Math.abs(s.hashCode() % shardingCount);
                String hashKey = String.format("%s$%s", key, num);
                tmpMap.computeIfAbsent(hashKey, k -> new ArrayList<>()).add(s);
            }
        }
        return tmpMap;
    }

    public void listSet(String key, List list) {
        if (!CollectionUtils.isEmpty(list)) {
            redisTemplate.opsForList().rightPushAll(key, list);
        }
    }

    public long listSize(String key) {
        return redisTemplate.opsForList().size(key);
    }

    public <T> Map<String, List<T>> listGet(String prefix, TypeReference<T> typeReference) {
        Map<String, List<T>> result = new ConcurrentHashMap<>();
        Set<String> setKeys = redisTemplate.keys(prefix + "*");
        setKeys.parallelStream().forEach(hashKey -> {
            try {
                List<T> list = listGet(typeReference, hashKey);
                result.put(hashKey, list);
            } catch (Exception e) {
                logger.error("获取hashKey为" + hashKey + "的redis缓存列表失败", e);
            }
        });
        return result;
    }

    public <T> List<T> listGet(TypeReference<T> typeReference, String key) {
        List<T> resultList = new LinkedList<>();
        long size = redisTemplate.opsForList().size(key);
        for (int index = 0; index < size; index = index + 1000) {
            List<Object> objList = new ArrayList<>();
            try {
                objList = redisTemplate.opsForList().range(key, index, 1000);
                objList.forEach(obj->{
                    resultList.add(objectMapper.convertValue(obj, typeReference));
                });
            } catch (Exception e) {
                logger.error(ExceptionUtils.getStackTrace(e));
            }
        }
        return resultList;
    }

    public void listDeleteOneList(String key) {
        deleteKey(key);
    }


    public Long listDel(String key, long count, Object value) {
        return redisTemplate.opsForList().remove(key, count, value);
    }

    public boolean deleteKey(String key) {
        return redisTemplate.delete(key);
    }

    public Set<String> getKeys(String pattern) {
        return getKeys(pattern, 10, TimeUnit.SECONDS);
    }

    public Set<String> getKeys(String pattern, int timeout, TimeUnit timeUnit) {
        Set<String> keys = new HashSet<>();
        CompletableFuture<Set<String>> getRedisKeysFuture = CompletableFuture.supplyAsync(() -> {
            try {
                return redisTemplate.keys(pattern);
            } catch (Exception e) {
                logger.error("获取redis中包含{}的key失败:", pattern, e);
                return null;
            }
        }).whenComplete((result, ex) -> {
            if (result != null) {
                keys.addAll(result);
            }
        });
        try {
            getRedisKeysFuture.get(timeout, timeUnit);
        } catch (Exception e) {
            logger.error("获取redis中包含{}的key超时:", pattern, e);
        }
        return keys;

    }

    public void strSet(String key, String value) {
        redisTemplate.opsForValue().set(key, value);
    }

    public void strSet(String key, String value, long timeout, TimeUnit timeUnit) {
        redisTemplate.opsForValue().set(key, value, timeout, timeUnit);
    }


    public String strGet(String key) {
        Object object = redisTemplate.opsForValue().get(key);
        if (object != null) {
            return object.toString();
        }
        return null;
    }

    public void expire(String key, long timeout, TimeUnit unit) {
        redisTemplate.expire(key, timeout, unit);
    }

    public void hLSetToOneGroup(String key, List<Object> list) {
        if (!CollectionUtils.isEmpty(list)) {
            redisTemplate.opsForList().leftPushAll(key, list);
        }
    }

    public <T> List<T> hLGetFromOneGroup(TypeReference<T> typeReference, String key) {
        List<T> result = new ArrayList<>();
        if (StringUtils.isNotBlank(key)) {
            long size = Optional.ofNullable(redisTemplate.opsForList().size(key)).orElse(0L);
            List<Object> objects = redisTemplate.opsForList().range(key, 0, size);
            if (!CollectionUtils.isEmpty(objects)) {
                for (Object o : objects) {
                    result.add(o != null ? objectMapper.convertValue(o, typeReference) : null);
                }
            }
        }
        return result;
    }

    public void setTypeAdd(String key, Object value) {
        redisTemplate.opsForSet().add(key, value);
    }

    public void setTypeAddAll(String key, Set<Object> value) {
        redisTemplate.opsForSet().add(key, value.toArray(new Object[1]));
    }

    public Set setTypeGet(String key) {
        return redisTemplate.opsForSet().members(key);
    }


    public boolean setTypeContain(String key, String value) {
        return redisTemplate.opsForSet().isMember(key, value);
    }
}

4、Redis链接配置工具类


import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
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.serializer.Jackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.StringRedisSerializer;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;

@Configuration
@ConditionalOnProperty(name = "spring.redis.host")
public class RedisConfiguration {

    @Value("${spring.redis.host}")
    private String host ;

    @Value("${spring.redis.port}")
    private int port ;

    @Value("${spring.redis.password}")
    private String password ;

    @Value("${spring.redis.timeout:3000}")
    private int timeout ;

    @Value("${redis.jedisPoolConfig.maxActive:300}")
    private int maxActive ;

    @Value("${redis.jedisPoolConfig.maxIdle:100}")
    private int maxIdle ;

    @Value("${redis.jedisPoolConfig.minIdle:1}")
    private int minIdle ;

    @Value("${redis.jedisPoolConfig.maxWait:5000}")
    private int maxWait ;

    @Value("${spring.redis.database:0}")
    private int database;

    @Bean
    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory){
        RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>();
        redisTemplate.setConnectionFactory(redisConnectionFactory);
        Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<>(Object.class);
        ObjectMapper objectMapper = new ObjectMapper();
//        objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
//        objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
        StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();
        redisTemplate.setKeySerializer(stringRedisSerializer);
        redisTemplate.setHashKeySerializer(stringRedisSerializer);
        redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
        redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer);
        redisTemplate.afterPropertiesSet();
        return redisTemplate;
    }

    @Bean
    public JedisPool JedisPoolFactory() {
        JedisPoolConfig poolConfig = new JedisPoolConfig();
        poolConfig.setMaxTotal(maxActive);
        poolConfig.setMaxIdle(maxIdle);
        poolConfig.setMinIdle(minIdle);
        poolConfig.setMaxWaitMillis( maxWait);
        JedisPool jedisPool =  new JedisPool(poolConfig,host,port,timeout,password,database);
        return jedisPool;
    }
}

5、配置文件内容

spring:
  redis:
    host: localhost
    port: 6379
    #password:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

菜鸟进军大神陆

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值