秒杀系统(三)Redis缓存和分布式锁

秒杀系统03——多级缓存

缓存:

后台问题描述

访问数据库查询商品信息:

读多写少

前端静态化页面:

上篇通过我们对数据进行静态化,也是有很多问题的,比如我们商品如果过多,freemark模板一定修改之后,我们所有的商品都需要重新再次生产静态化,这个工作量实在是太大了。

引入缓存:

缓存作用:

秒杀系统(三)Redis缓存和分布式锁_第1张图片

Redis缓存实战:

加入Redis

/**
     * 获取商品详情信息  加入redis
     *
     * @param id 产品ID
     */
    public PmsProductParam getProductInfo2(Long id) {
        PmsProductParam productInfo = null;
        //从缓存Redis里找
        productInfo = redisOpsUtil.get(RedisKeyPrefixConst.PRODUCT_DETAIL_CACHE + id, PmsProductParam.class);
        if (null != productInfo) {
            return productInfo;
        }
        productInfo = portalProductDao.getProductInfo(id);
        System.out.println("我被执行了");
        if (null == productInfo) {
            log.warn("没有查询到商品信息,id:" + id);
            return null;
        }
        checkFlash(id, productInfo);
        redisOpsUtil.set(RedisKeyPrefixConst.PRODUCT_DETAIL_CACHE + id, productInfo, 3600, TimeUnit.SECONDS);
        return productInfo;
    }

上述方案的问题: 数据拷贝、数据缓存的共性问题

1、数据一致性的问题,怎么保证Redis和mysql数据一致性的问题

Jmeter 1000次请求图,吞吐量630.5/sec

秒杀系统(三)Redis缓存和分布式锁_第2张图片

数据一致性的两种解决方案:

1、最终一致性方案:

设置超时时间来解决。设置Redis缓存数据的超时时间。Redis过期,下次请求还是得从mysql中获取

redisOpsUtil.set(RedisKeyPrefixConst.PRODUCT_DETAIL_CACHE+id,productInfo,360,TimeUnit.SECONDS);

2、实时一致性方案:

交易canal binlog

两个问题(高并发)

压缩的问题 、减少内存

问题1,Redis缓存的对象有时候很大,Redis只解决了磁盘IO,没有解决网络IO问题。网络传输IO依旧很大。

问题2,Redis数据会占用大量内存,采用过期机制可以缓解内存压力。

网络I/O解决方法,压缩对象内容,采用序列化和压缩方法

压力测试情景

当前在jmeter里面进行1000个并发请求商品id=26,结果如下图

秒杀系统(三)Redis缓存和分布式锁_第3张图片

跟预期只set一次redis 是有出入,为何会这样子了?

re: 是因为出现了并发问题

当我第二次再去访问,此时此刻没有日志输出,说明全部走了缓存

并发问题:并发编程》并发问题》可以用锁的方式来实现 java并发,但是加锁的方式(不适合分布式场景)

加锁的方式为什么不适用分布式场景:

情景: 假如有800个请求,400个去访问一个服务器,400个去访问另一个服务器。

其中加锁 synchronized 可以保证其中一个jvm的400个请求是串行去访问。

但是两个服务器访问数据库的顺序不能保证,是否是跨进程访问。

上锁时间、和代码块的运行时间

解决方法:采用分布式锁:redis、zookeeper

zookeeper 有强认证,适合分布式

Redis分布式实战

/*
 * zk分布式锁
 */
@Autowired
private ZKLock zkLock;
private String lockPath = "/load_db";

@Autowired
RedissonClient redission;

/**
     * 获取商品详情信息  加入redis 加入锁
     *
     * @param id 产品ID
     */
public PmsProductParam getProductInfo3(Long id) {
    PmsProductParam productInfo = null;
    //从缓存Redis里找
    productInfo = redisOpsUtil.get(RedisKeyPrefixConst.PRODUCT_DETAIL_CACHE + id, PmsProductParam.class);
    if (null != productInfo) {
        return productInfo;
    }
    RLock lock = redission.getLock(lockPath + id); //获取重入锁
    try {
        if (lock.tryLock()) { //获取锁-同一个线程可重入
            productInfo = portalProductDao.getProductInfo(id);
            if (null == productInfo) {
                log.warn("没有查询到商品信息,id:" + id);
                return null;
            }
            checkFlash(id, productInfo);
            redisOpsUtil.set(RedisKeyPrefixConst.PRODUCT_DETAIL_CACHE + id, productInfo, 3600, TimeUnit.SECONDS);
        } else {
            productInfo = redisOpsUtil.get(RedisKeyPrefixConst.PRODUCT_DETAIL_CACHE + id, PmsProductParam.class);
        }
    } finally {
        if (lock.isLocked()) {
            if (lock.isHeldByCurrentThread())
                lock.unlock();
        }
    }
    return productInfo;
}

ps:重入锁和偏向锁的定义

重入锁,可重入性是指比如一个线程获得了对象A上的锁,如果它第二次请求A的锁必然可以获得(也就是说不会自己把自己锁住),可重入性是线程必须满足的,不然很多代码就会死锁了

偏向锁,偏向锁是说如果线程请求一个自己已经获得的锁,它不会去再次执行lock和unlock,这样可以提升性能。
如何实现可重入都是一样的,就是把锁的拥有者记下来,当申请锁的时候看一下锁是否已经被占有了,如果有人占着锁,看看是不是就是申请者自己。

可以看到使用了Redis锁,QPS立马提升了很多

秒杀系统(三)Redis缓存和分布式锁_第4张图片


Redis自身提供的锁 setnx,使用setnx进行并发锁进行

缓存应用场景:

1、访问量大、QPS高、更新频率不是很高的业务

2、数据一致性要求不高

Redis缓存问题

缓存击穿问题(热点数据单个key):

对于一些设置了过期时间的key,如果这些key可能会在某些时间点被超高并发地访问,是一种非常“热点”的数据。这个时候,需要考虑一个问题:缓存被“击穿”的问题,这个和缓存雪崩的区别在于这里针对某一key缓存,前者则是很多key,缓存击穿是对一个key一段时间其中进行访问。

秒杀系统(三)Redis缓存和分布式锁_第5张图片

解决方案:

1.加锁,在未命中缓存时,通过加锁避免大量请求访问数据库。通过加锁

2.不允许过期。物理不过期,也就是不设置过期时间。而是逻辑上定时在后台异步的更新数据。最简单粗暴的情况

3.采用二级缓存。L1缓存失效时间短,L2缓存失效时间长。请求优先从L1缓存获取数据,如果未命中,则加锁,保证只有一个线程去数据库中读取数据然后再更新到L1和L2中。然后其他线程依然在L2缓存获取数据。采用二级缓存,不可能两个Redis都被击穿。

缓存穿透问题(恶意攻击、访问不存在数据):

缓存穿透是指查询一个一定不存在的数据,由于缓存是不命中时被动写的,并且出于容错考虑,如果从存储层查不到数据则不写入缓存,这将导致这个不存在的数据每次请求都要到存储层去查询,失去了缓存的意义。在流量大时,可能DB就挂掉了,要是有人利用不存在的key频繁攻击我们的应用,这就是漏洞

有人采用不存在的key频繁的去访问,就会发生缓存穿透的问题

秒杀系统(三)Redis缓存和分布式锁_第6张图片

**解决方案:**有很多种方法可以有效地解决缓存穿透问题

1、最常见的则是采用布隆过滤器,将所有可能存在的数据哈希到一个足够大的bitmap中,一个一定不存在的数据会被这个bitmap拦截掉,从而避免了对底层存储系统的查询压力。采用布隆过滤器

2、另外也有一个更为简单粗暴的方法(我们采用的就是这种),如果一个查询返回的数据为空(不管是数据不存在,还是系统故障),我们仍然把这个空结果进行缓存,但它的过期时间会很短,最长不超过五分钟。 直接将空节点也进行缓存到Redis中。

秒杀系统(三)Redis缓存和分布式锁_第7张图片

缓存雪崩(同一时间失效,并发量大):

雪崩是指在我们设置缓存时采用了相同的过期时间,导致缓存在某一时刻同时失效,请求全部转发到DB,DB瞬时压力过重雪崩。

大量的Redis同一时间失效,所有的接口瞬间访问数据库。

秒杀系统(三)Redis缓存和分布式锁_第8张图片

秒杀系统(三)Redis缓存和分布式锁_第9张图片

解决方案:

**1、**缓存失效时的雪崩效应对底层系统的冲击非常可怕。大多数系统设计者考虑用加锁或者队列的方式保证缓存的单线程(进程)写,从而避免失效时大量的并发请求落到底层存储系统上。这里分享一个简单方案就是将缓存失效时间分散开,比如我们可以==在原有的失效时间基础上增加一个随机值==,比如1-5分钟随机,这样每一个缓存的过期时间的重复率就会降低,就很难引发集体失效的事件。

2、事前:这种方案就是在发生雪崩前对缓存集群实现高可用,如果是使用 Redis,可以使用 主从+哨兵 ,Redis Cluster 来避免 Redis 全盘崩溃的情况。

3、事中:使用 Hystrix进行限流 & 降级 ,比如一秒来了5000个请求,我们可以设置假设只能有一秒 2000个请求能通过这个组件,那么其他剩余的 3000 请求就会走限流逻辑。然后去调用我们自己开发的降级组件(降级),比如设置的一些默认值呀之类的。以此来保护最后的 MySQL 不会被大量的请求给打死。采用限流&降级

4、事后:开启Redis持久化机制,尽快恢复缓存集群。发生雪崩后,快速恢复文件

缓存和数据库双写一致性问题:

一致性问题是分布式常见问题,还可以再分为最终一致性和强一致性。数据库和缓存双写,就必然会存在不一致的问题。

re: 先明白一个前提。就是如果对数据有强一致性要求,不能放缓存。

我们所做的一切,只能保证最终一致性。另外,我们所做的方案其实从根本上来说,只能说降低不一致发生的概率,无法完全避免。因此,有强一致性要求的数据,不能放缓存。

引入分布式锁代码:

1、引入redis依赖

<!--加入redisson-->
<dependency>
    <groupId>org.redisson</groupId>
    <artifactId>redisson</artifactId>
    <version>3.6.5</version>
</dependency>

@Bean
public RedissonClient redissonClient(){
    Config config = new Config();
    config.useSingleServer().setAddress("redis://tlshop.com:6379").setPassword("123456").setDatabase(1);
    return Redisson.create(config);
}

单机加锁:

秒杀系统(三)Redis缓存和分布式锁_第10张图片

分布式加锁:

同一时间同一个数据请求过来,比如set 100 value:(1 2 3)

秒杀系统(三)Redis缓存和分布式锁_第11张图片

使用ZooKeeper分布式锁实战

/**
 * 获取商品详情信息
 *
 * @param id 产品ID
 */
public PmsProductParam getProductInfo(Long id) {
    PmsProductParam productInfo = null;
    productInfo = redisOpsUtil.get(RedisKeyPrefixConst.PRODUCT_DETAIL_CACHE + id, PmsProductParam.class);
    if (productInfo != null) {
        log.info("get redis productId:" + productInfo);
        return productInfo;
    }
    try {
        if (zkLock.lock(lockPath + "_" + id)) {
            productInfo = portalProductDao.getProductInfo(id);
            if (null == productInfo) {
                return null;
            }
            checkFlash(id, productInfo);
            log.info("set db productId:" + productInfo);
            //缓存失效时间随机
            redisOpsUtil.set(RedisKeyPrefixConst.PRODUCT_DETAIL_CACHE + id, productInfo, 3600, TimeUnit.SECONDS);
        } else {
            //问题:返回为null
            productInfo = redisOpsUtil.get(RedisKeyPrefixConst.PRODUCT_DETAIL_CACHE + id, PmsProductParam.class);
        }
    } finally {
        log.info("unlock :" + productInfo);
        zkLock.unlock(lockPath + "_" + id);
    }
    return productInfo;
}

private void checkFlash(Long id, PmsProductParam productInfo) {
    FlashPromotionParam promotion = flashPromotionProductDao.getFlashPromotion(id);
    if (!ObjectUtils.isEmpty(promotion)) {
        productInfo.setFlashPromotionCount(promotion.getRelation().get(0).getFlashPromotionCount());
        productInfo.setFlashPromotionLimit(promotion.getRelation().get(0).getFlashPromotionLimit());
        productInfo.setFlashPromotionPrice(promotion.getRelation().get(0).getFlashPromotionPrice());
        productInfo.setFlashPromotionRelationId(promotion.getRelation().get(0).getId());
        productInfo.setFlashPromotionEndDate(promotion.getEndDate());
        productInfo.setFlashPromotionStartDate(promotion.getStartDate());
        productInfo.setFlashPromotionStatus(promotion.getStatus());
    }

发现在高并发下增加了分布式锁可以解决刚才那个问题,但是也降低了qps,所以这儿还是要根据需求而定

分布式锁原理:

zk图

秒杀系统(三)Redis缓存和分布式锁_第12张图片

总结:

1、线程来会直接创建一个锁节点下的下一个临时顺序节点

2、如果自己不是第一个节点,就给自己加上一个节点监听器

3、只要上一个节点释放锁,自己就排到前面,相当于一个排队机制

你可能感兴趣的:(缓存,redis,分布式,java,分布式锁)