Redis最快速入门!缓存更新策略(超详细)

声明

本文是笔者学习黑马课程做的笔记

Redis实现缓存

为什么要使用缓存

缓存数据存储于代码中,而代码运行在内存中,内存的读写性能远高于磁盘,缓存可以大大降低用户访问并发量带来的服务器读写压力

实际开发过程中,企业的数据量,少则几十万,多则几千万,这么大数据量,如果没有缓存来作为"避震器",系统是几乎撑不住的,所以企业会大量运用到缓存技术;

但是缓存也会增加代码复杂度和运营的成本:

如何使用缓存

实际开发中,会构筑多级缓存来使系统运行速度进一步提升,例如:本地缓存与redis中的缓存并发使用

浏览器缓存:主要是存在于浏览器端的缓存

应用层缓存:可以分为tomcat本地缓存,比如之前提到的map,或者是使用redis作为缓存

数据库缓存:在数据库中有一片空间是 buffer pool,增改查数据都会先加载到mysql的缓存中

CPU缓存:当代计算机最大的问题是 cpu性能提升了,但内存读写速度没有跟上,所以为了适应当下的情况,增加了cpu的L1,L2,L3级的缓存

添加商户缓存

查询商品页信息时是直接查询数据库的,我们可以使用redis来缓存数据提高页面查询速度

@Autowired
private StringRedisTemplate redisTemplate;
@Override
public Result queryById(Long id) {
//首先查询redis有没有这个缓存
String key=CACHE_SHOP_KEY + id;
String strJson = redisTemplate.opsForValue().get(key);
//有则直接返回
if (StrUtil.isNotBlank(strJson)){
    Shop shop = JSONUtil.toBean(strJson, Shop.class);
    return Result.ok(shop);
}
//没有则查询数据库有没有这个数据,没有则返回错误信息
Shop shop = getById(id);
if (shop==null){
    return Result.fail("没有该商铺信息");
}
//有的话就存入redis作为缓存使用,将对象转为Json数据
redisTemplate.opsForValue().set( key,JSONUtil.toJsonStr(shop));
return Result.ok(shop);
}

缓存更新策略

缓存更新是redis为了节约内存而设计出来的一个东西,主要是因为内存数据宝贵,当我们向redis插入太多数据,此时就可能会导致缓存中的数据过多,所以redis会对部分数据进行更新,或者把他叫为淘汰更合适。

内存淘汰:redis自动进行,当redis内存达到咱们设定的max-memery的时候,会自动触发淘汰机制,淘汰掉一些不重要的数据(可以自己设置策略方式)

超时剔除:当我们给redis设置了过期时间ttl之后,redis会将超时的数据进行删除,方便咱们继续使用缓存

主动更新:我们可以手动调用方法把缓存删掉,通常用于解决缓存和数据库不一致问题

根据业务需求来选择

数据库缓存不一致解决方案:

由于我们的缓存的数据源来自于数据库,而数据库的数据是会发生变化的,因此,如果当数据库中数据发生变化,而缓存却没有同步,此时就会有一致性问题存在,其后果是:

用户使用缓存中的过时数据,就会产生类似多线程数据安全问题,从而影响业务,产品口碑等;怎么解决呢?有如下几种方案

Cache Aside Pattern 人工编码方式:缓存调用者在更新完数据库后再去更新缓存,也称之为双写方案

Read/Write Through Pattern : 由系统本身完成,数据库与缓存的问题交由系统本身去处理

Write Behind Caching Pattern :调用者只操作缓存,其他线程去异步处理数据库,实现最终一致

总结

缓存更新策略的最佳实践方案:

1.低一致性需求:使用Redis自带的内存淘汰机制

2.高一致性需求:主动更新,并以超时剔除作为兜底方案

读操作:

•缓存命中则直接返回

•缓存未命中则查询数据库,并写入缓存,设定超时时间

写操作:

•先写数据库,然后再删除缓存

先写数据库再删除缓存发生的概率小

要确保数据库与缓存操作的原子性

给查询商铺的缓存设置超时时间和修改商户商铺那里增加删除缓存

@Transactional
public Result update(Shop shop) {
if (shop.getId()==null){
    return Result.fail("id不能为空");
}
//修改数据库
updateById(shop);
//删除缓存
String key=CACHE_SHOP_KEY + shop.getId();
redisTemplate.delete(key);
return Result.ok();
}
  • 5
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值