关于Redission读写锁

本文介绍了读写锁在并发场景中的使用,包括其类型(共享锁和排它锁),以及如何解决修改数据时的并发问题。通过引入延迟队列,即使在写入期间无法获取读锁,也能保证统计监控数据的准确性。文中展示了在Redisson中使用读写锁的实际代码示例和使用延迟队列进行优化的方法。
摘要由CSDN通过智能技术生成

目录

什么是读写锁:

读写锁分两种:

1.读锁(共享锁):

2.写锁(排它锁):

优点:

仍有问题:

改进:

读写锁使用:

修改时:

读取信息时:

生产者:

消费者:


思考一个场景,当我们的一个数据正在修改的时候,正好有用户来访问,我们对这个数据的相关统计监控数据就无法正确进行。

什么是读写锁:

读写锁是一种用于管理对共享资源的访问的同步机制,允许多个线程同时读取共享资源,但在写入时保证独占访问,以确保数据的一致性和完整性。

读写锁分两种:
1.读锁(共享锁):

多个线程可以同时获取读锁,用于并发读取共享资源。读锁在没有写锁的情况下可以被多个线程持有。

2.写锁(排它锁):

写锁是独占的,一旦一个线程获取了写锁,其他线程无法同时获取读锁或者写锁。写锁用于修改共享资源,确保在写入时没有其他线程能有访问

优点:

读写锁的优势在于它允许多个线程同时读取共享资源,提高了读取的并发性,从而提升了性能。但是,在写入时必须独占资源,以确保数据的一致性。这种锁的使用场景适用于读操作频繁,写操作较少的情况。

仍有问题:

当我们在修改数据的时候拿到了写锁,读锁获取不到,难道这段时间内用户一直不允许访问么?

如果获取不到就一直等着,会有很多线程堆在这里,应用会垮掉。

如果获取不到就返回,那我们后续的数据统计监控,记录信息就不精准

改进:

我们用到了延迟队列:

当我们获取读锁失败的时候,把东西扔给延迟队列,可以设置一个等待时间,时间到了再去统计,如果还没获取到锁,那就再扔进去,等待一定时间后统计。

读写锁使用:
修改时:
            //拿到写锁
            RReadWriteLock readWriteLock = redissonClient.getReadWriteLock(String.format(LOCK_GID_UPDATE_KEY, requestParam.getFullShortUrl()));
            RLock rLock = readWriteLock.writeLock();
            if (!rLock.tryLock()) {
                throw new ServiceException("短链接正在被访问,请稍后再试...");
            }
            try {
            } 
            finally {
            rLock.unlock();
            }
读取信息时:
        RReadWriteLock readWriteLock = redissonClient.getReadWriteLock(String.format(LOCK_GID_UPDATE_KEY, fullShortUrl));
        //先拿读锁
        RLock rLock = readWriteLock.readLock();
        if (!rLock.tryLock()) {
            delayShortLinkStatsProducer.send(statsRecord);
            return;
        }
         try {
         } 
         catch (Throwable ex) {
            throw new ServiceException("监控器异常");
        }
         finally {
         rLock.unlock();
         }
生产者:
public class DelayShortLinkStatsProducer {

    private final RedissonClient redissonClient;

    /**
     * 发送延迟消费短链接统计
     *
     * @param statsRecord 短链接统计实体参数
     */
    public void send(ShortLinkStatsRecordDTO statsRecord) {
        RBlockingDeque<ShortLinkStatsRecordDTO> blockingDeque = redissonClient.getBlockingDeque(DELAY_QUEUE_STATS_KEY);
        RDelayedQueue<ShortLinkStatsRecordDTO> delayedQueue = redissonClient.getDelayedQueue(blockingDeque);
        delayedQueue.offer(statsRecord, 5, TimeUnit.SECONDS);
    }
}
消费者:
@Component
@RequiredArgsConstructor
public class DelayShortLinkStatsConsumer implements InitializingBean {

    private final RedissonClient redissonClient;
    private final ShortLinkService shortLinkService;

    public void onMessage() {
        Executors.newSingleThreadExecutor(
                        runnable -> {
                            Thread thread = new Thread(runnable);
                            thread.setName("delay_short-link_stats_consumer");
                            thread.setDaemon(Boolean.TRUE);
                            return thread;
                        })
                .execute(() -> {
                    RBlockingDeque<ShortLinkStatsRecordDTO> blockingDeque = redissonClient.getBlockingDeque(DELAY_QUEUE_STATS_KEY);
                    RDelayedQueue<ShortLinkStatsRecordDTO> delayedQueue = redissonClient.getDelayedQueue(blockingDeque);
                    for (; ; ) {
                        try {
                            ShortLinkStatsRecordDTO statsRecord = delayedQueue.poll();
                            if (statsRecord != null) {
                                shortLinkService.shortLinkStats(null, null, statsRecord);
                                continue;
                            }
                            LockSupport.parkUntil(500);
                        } catch (Throwable ignored) {
                        }
                    }
                });
    }

    @Override
    public void afterPropertiesSet() throws Exception {
        onMessage();
    }
}

后续我们可以使用Rabbit或者别的MQ的延迟队列来进行更好的替换优化

  • 12
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Redisson提供了读写锁的功能。读写锁在并发场景中非常有用,特别是对于读操作远多于写操作的情况。读写锁允许多个线程同时持有读锁,但只允许一个线程持有写锁。 通过使用Redisson的读写锁,你可以实现对某个关键字进行并发的读和写操作控制。当一个线程持有读锁时,其他线程也可以持有读锁,但是当有线程持有写锁时,其他的写锁和读锁都会被阻塞。 在实际应用中,如果有一个场景需要并发读取数据,然后并发进行扣减操作,可能会出现扣减失败的情况。这是因为在读取的时候还有库存,但是在扣减的时候库存已经没有了。为了避免这种情况,可以使用Redisson的读写锁来保证并发读写的线程安全性。 在使用Redisson的读写锁时,可以通过以下方式进行操作: - 使用`readLock()`方法获取读锁,并通过`lock()`方法加锁,然后使用`unlock()`方法释放读锁。 - 使用`writeLock()`方法获取写锁,并通过`lock()`方法加锁,然后使用`unlock()`方法释放写锁。 在你的代码示例中,可以看到使用Redisson的读写锁的具体操作。通过`getReadWriteLock(key)`方法获取读写锁,然后使用`readLock()`方法获取读锁或使用`writeLock()`方法获取写锁。最后使用`lock()`方法加锁,使用`unlock()`方法释放锁。 总结起来,Redisson的读写锁提供了一种并发控制机制,可以在并发读写场景下确保数据的一致性与线程安全性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [redission读写锁解决db和缓存双写不一致](https://blog.csdn.net/weixin_43944305/article/details/120191365)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值