Redis如何实现原子性自增自减

这篇具有很好参考价值的文章主要介绍了Redis如何实现原子性自增自减。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、背景        

假设有一个需求,包含简单的两个步骤:

  1. 第一步是用户获取验证码,检验验证码成功后跳转到表单填写页面;
  2. 第二步是用户填写表单并提交申请

为了防止用户跳过第一步直接提交申请,我们采取了以下策略:

  1. 在验证码验证成功后,我们将用户Id作为Key,剩余可申请次数作为Value,将这个键值对存储在Redis中并设置过期时间;
  2. 在收到用户提交的申请请求后,首先检查Redis中该用户Id对应的Value是否存在且大于0。我们将申请次数减一,并允许用户提交申请;如果不满足条件,则直接抛出异常。

二、increment

针对以上策略的第一步,直接上伪代码

1.检验验证码方法如下:

//以下为伪代码,重点关注redis的调用逻辑

//校验验证码方法
public void checkCode(String code){

    //校验验证码逻辑
    ...

    
    //校验通过
    redisBuryingPoint(userId);

}

以上代码,略过了校验验证码的逻辑,在验证码校验成功后,调用了'redisBuryingPoint(userId)'方法进行埋点,其中userId为用户Id。

2.redisBuryingPoint方法如下:

/**
 * redis埋点
 * @param userId 用户id
*/
public void redisBuryingPoint(String userId) {

    int value = redisService.increment(userId, 1 * 60 * 60L);
    logger.info("用户[{}]验证成功共计[{}]次", userId, value);
}

3.redisService#increment(String key, Long expirationTimeInSeconds) 方法如下:

/**
 * 自增  
 * @param key 要加一的键
 * @param expirationTimeInSeconds 过期时间
*/
public int increment(String key, Long expirationTimeInSeconds) {
        // +1 操作
        int result = redisTemplate.opsForValue().increment(key).intValue();
        // 设置过期时间
        redisTemplate.expire(KEY, expirationTimeInSeconds, TimeUnit.SECONDS);
        //返回
        return result;
}

4.小结

以上就完成了redis埋点的过程。即使同一个用户在同一时间进行多个验证码的验证操作,即在并发场景下多个请求同时调用'increment() '方法时,RedisTemplate 会自动处理并发操作,确保操作的原子性一致性

三、decrement

针对以上策略的第二步,伪代码如下

1.申请提交预校验方法如下:

/**
 * 预校验:校验短信验证成功的次数
 * @param userId 用户Id
*/
private void preCheck(String userId) {

    Long result = redisService.decrement(userId);

    if (result != null) {
        // 成功递减
        logger.info("发起申请成功,用户[{}]验证成功次数剩余[{}]次", userId, result);
    } else {
        // 值不存在,无需递减操作
        logger.error("发起申请失败,用户[{}]验证成功次数不足", userId);
        throw new RuntimeException("发起申请失败,请先进行短信验证!")
    }
}

在上述代码中,我们直接调用redisService的 decrement() 方法,我们检查返回的结果是否为 null,如果不为 null,表示递减操作成功,并打印递减后的剩余次数。如果结果为 null,表示值不存在,没有进行递减操作,打印错误日志并抛出异常。

2.redisService#decrement(String key) 方法如下:

private RedisScript<Long> decrementScript;

public RedisService() {
        //定义 Lua 脚本
        this.decrementScript = new DefaultRedisScript<>(
                "if redis.call('exists', KEYS[1]) == 1 and tonumber(redis.call('get', KEYS[1])) > 0 then " +
                "   return redis.call('decr', KEYS[1]) " +
                "else " +
                "   return nil " +
                "end",
                Long.class);
    }


/**
 * 自减 
 * @param key 要减一的键
 * 
*/
public Long decrement(String key) {
        return redisTemplate.execute(decrementScript, Collections.singletonList(key));    
}

在上述代码中,我们使用 Lua 脚本来执行递减操作。脚本首先检查键是否存在,如果存在则执行递减操作,如果不存在则返回空。在 Java 代码中,我们使用 DefaultRedisScript 来定义 Lua 脚本,并通过 redisTemplate.execute() 方法来执行脚本。

3.小结

以上就完成了redis消费的过程。即使同一个用户在同一时间提交多次申请,即在并发场景下多个请求同时调用'decrement()'方法时,Redis 执行 Lua 脚本,会将整个脚本作为一个原子操作进行执行。确保操作的原子性一致性

四、其他

        当 Redis 的键不存在时,使用 opsForValue().increment() 方法会将键的值初始化为 1,并执行递增操作;使用 opsForValue().decrement() 方法会将键的值初始化为 -1,并执行递减操作。

        所以,如果在调用 increment() 方法时,键对应的值不存在,它将被赋值为 1;如果在调用 decrement() 方法时,键对应的值不存在,它将被赋值为 -1。

    文章来源地址https://www.toymoban.com/news/detail-604265.html

到了这里,关于Redis如何实现原子性自增自减的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • springboot整合redis+lua实现getdel操作保证原子性

    原始代码 脚本逻辑先获取redis的值,判断是否等于期望值。 条件成立则删除,不成立则返回0 测试代码 根据上面的逻辑加了测试, 在判断成功后等待5秒后执行删除操作。同时开启另外一个线程去修改这个key的值, 发现修改的线程一直阻塞。直到等待的线程5秒结束后且完成

    2024年02月05日
    浏览(48)
  • 基于 Redis + Lua 脚本实现分布式锁,确保操作的原子性

    1.加锁的Lua脚本: lock.lua 2.解锁的Lua脚本: unLock.lua 3.将资源文件放在资源文件夹下 4.Java中调用lua脚本 1)获取文件方式 2)lua字符串方式 5.jedis调用Lua脚本实现分布式重试锁 1)引入jedis依赖 2)jedis调用lua

    2024年02月07日
    浏览(56)
  • Redis自增生成

    Redis 是一个开源的内存数据结构存储系统,可以用来作为数据库、缓存和消息中间件。Redis 的特点是高性能、可扩展性强,支持多种数据结构等。在使用 Redis 时,常常需要用到自增 ID 的功能,例如生成订单 ID 等。本文将介绍如何使用 Java 实现 Redis 自增生成 ID 的功能。 在

    2024年02月12日
    浏览(35)
  • Redis实现分布式锁之----超时和失效(非原子性)问题----解决方案

    Redis实现分布式锁之----超时和失效(非原子性)问题----解决方案 超时和失效(非原子性)问题 原子性问题 :上锁时存入线程名称,删除时要先判断锁内的名称是不是自己的,是再删除,但是后面的判断 和删除非原子性 ,会有并发安全问题。 不可重入问题 :一个线程只能

    2024年02月07日
    浏览(43)
  • 【Redis】Redis事务:原子性与回滚的真相揭秘

    大家好,我是mep。今天一起来探讨一下Redis缓存的问题,SpringBoot如何集成Redis网上文章很多,基本都是介绍如何配置redisTemplate,如何调用,本文就不过多介绍了。这次我们研究的是:Redis的事务。 首先抛出一个问题,Redis支持事务吗?     答案肯定是 支持 ,不然也不需要我

    2024年02月08日
    浏览(44)
  • 【Redis】5、Redis 的分布式锁、Lua 脚本保证 Redis 命令的原子性

    🎄 分布式锁:满足分布式系统或集群模式下 多进程可见 并且 互斥的 锁 🎄 分布式锁的核心是实现多进程之间锁的互斥 ,而满足这一点的方式有很多,常见的有三种: 🎄锁获取了,还没有来得及设置过期时间服务器就宕机了 🎄保证 setnx(获取锁)和 expire 设置过期时间两

    2024年02月15日
    浏览(40)
  • Redis的内存淘汰策略有哪些?Redis的发布订阅功能是如何实现的?如何监控Redis的性能?Redis的并发竞争问题如何解决?

    Redis的内存淘汰策略有以下几种: noeviction :不进行任何内存淘汰,当内存用完时,新的写操作将会返回错误。 volatile-lru :在所有已设置过期时间的键中,使用近似LRU算法删除最长时间未使用的键,直到腾出足够的内存空间为止。 volatile-ttl :在所有已设置过期时间的键中,

    2024年02月12日
    浏览(83)
  • redis如何实现缓存预热

    在业务系统中,我们需要在程序启动的时候加载一些常用的数据到内存数据库中,从而减少业务数据库的压力。这就是我们常提到的缓存预热。官方一点的解释是这样的: 缓存预热 是一种在程序启动或缓存失效之后,主动将热点数据加载到缓存中的策略。这样,在实际请求

    2024年02月07日
    浏览(36)
  • Redis如何实现消息队列

    Redis可以通过List数据结构实现简单的消息队列。在Redis中,我们可以使用 LPUSH 命令将消息推送到列表的左侧,使用 RPOP 命令从列表的右侧获取消息。这样,就可以实现一个先进先出(FIFO)的消息队列。 下面是一个使用Redis实现消息队列的简单示例: 首先,确保你已经安装了

    2024年02月14日
    浏览(36)
  • redis — redis cluster集群模式下如何实现批量可重入锁?

    一、redis cluster 集群版 在Redis 3.0版本以后,Redis发布了Redis Cluster。该集群主要支持搞并发和海量数据处理等优势,当 Redis 在集群模式下运行时,它处理数据存储的方式与作为单个实例运行时不同。这是因为它应该准备好跨多个节点分发数据,从而实现水平可扩展性。具体能力表

    2024年01月21日
    浏览(46)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包