[Redis]黑马点评-商铺类型缓存使用List(练习题)

该代码实现了一个服务类,通过StringRedisTemplate操作Redis缓存获取商铺分类信息。如果缓存中存在数据则直接返回,否则从数据库查询并按排序返回,同时将查询结果存入缓存。
public class ShopTypeServiceImpl extends ServiceImpl<ShopTypeMapper, ShopType> implements IShopTypeService {
​​​​​​​
    @Resource
    private StringRedisTemplate stringRedisTemplate;

    @Override
    public Result getTypeList() {
        String key= RedisConstants.CACHE_SHOP_TYPE;
//        1.从缓存中获取商铺分类信息 range:获取一个指定范围的值
        List<String> tyepList = stringRedisTemplate.opsForList().range(key, 0, -1);
//        2.判断缓存中是否存在
        if (Objects.requireNonNull(tyepList).size() != 0) {
//            存在直接返回
            List<ShopType> types = new ArrayList<>();
            for (String type : tyepList) {
                ShopType shopType = JSONUtil.toBean(type, ShopType.class);
                types.add(shopType);
            }
           return Result.ok(types);
        }
//        不存在则查询数据
        List<ShopType> shopTypeList = this.query().orderByAsc("sort").list();
        if (shopTypeList == null) {
           return Result.fail("数据不存在!");
        }
//        将查询到数据添加到缓存中
        assert shopTypeList != null;
        for (ShopType type : shopTypeList) {
//           rightPush向一个集合右边添加数据
            stringRedisTemplate.opsForList().rightPush(key,JSONUtil.toJsonStr(type));
        }
//        返回数据
        return Result.ok(shopTypeList);
    }
}

### 使用 Redis 实现店铺类型数据的缓存 为了实现店铺类型的高效缓存并提升应用性能,可以采用以下策略: #### 方法设计 定义 `saveShopTypeToRedis` 函数用于保存店铺类型Redis 中,并设置合理的逻辑过期时间。此操作不仅能够减少数据库的压力,还能确保即使在高并发场景下也能提供稳定的服务。 ```java public void saveShopTypeToRedis(Long typeId, Long seconds) { // 查询店铺类型数据 ShopType shopTypeById = getShopTypeById(typeId); // 封装要存储的数据结构 RedisData redisData = new RedisData(); redisData.setData(shopTypeById); // 设置逻辑上的过期时间戳 LocalDateTime expireTime = LocalDateTime.now().plusSeconds(seconds); redisData.setExpireTime(expireTime); // 序列化对象为JSON字符串形式存入Redis String jsonStr = JSONUtil.toJsonStr(redisData); stringRedisTemplate.opsForValue().set( RedisConstants.CACHE_SHOP_TYPE_KEY + typeId, jsonStr ); } ``` 上述代码片段展示了如何将特定ID对应的店铺类型信息持久化至Redis中[^1]。 对于读取操作,则需先尝试从Redis获取对应键值;如果命中则解析返回给前端;未命中的情况下再回源查询真实库表记录,并更新最新版本到缓存层以便后续访问能快速响应。 ```java public ShopType queryWithLogicalExpire(Long id){ // 构造缓存Key String key = RedisConstants.CACHE_SHOP_TYPE_KEY + id; // 优先级最高的是直接从本地内存或更靠近用户的节点处取得所需资源 Object resultObj = localCache.getIfPresent(key); if (resultObj != null && !isExpired(resultObj)){ return ((RedisData<ShopType>)resultObj).getData(); } // 获取缓存中的json串 String typeJson = stringRedisTemplate.opsForValue().get(key); if(StrUtil.isBlank(typeJson)) { // 缓存穿透处理... return null; } // 反序列化成对象 RedisData redisData = JSONUtil.toBean(typeJson, RedisData.class); JSONObject data = JSONUtil.parseObj(redisData.getData()); ShopType shopType = BeanUtil.fillBeanWithData(data, new ShopType(), true); // 判断是否已过期 LocalDateTime expireTime = redisData.getExpireTime(); if(expireTime.isAfter(LocalDateTime.now())) { // 如果没过期就正常返回商品详情 return shopType; } else{ // 已经过期了,删除旧版缓存条目 stringRedisTemplate.delete(key); // 背景异步重构新缓存项 rebuildCacheOnBackground(id,key); // 返回null 或者默认值/降级方案 return null; } } private boolean isExpired(Object obj){ try{ RedisData<?> rd = (RedisData<?>)obj; return rd.getExpireTime().isBefore(LocalDateTime.now()); }catch(Exception e){ log.error("判断缓存是否过期失败",e); return false; } } ``` 这段Java程序实现了带有逻辑过期机制的商品详情页缓存功能,在保证高性能的同时也兼顾了一定程度上的一致性和安全性。 当遇到缓存雪崩风险时,可以通过引入随机延迟、限流措施以及分布式锁等方式来缓解突发流量冲击带来的负面影响[^4]。
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值