首页 > 数据库 >redis实际应用场景及并发问题的解决

redis实际应用场景及并发问题的解决

时间:2024-03-24 21:23:12浏览次数:29  
标签:回合 场景 return String redis private 金币 并发 id

业务场景
接下来要模拟的业务场景:

每当被普通攻击的时候,有千分之三的概率掉落金币,每回合最多爆出两个金币。

1.每个回合只有15秒。

2.每次普通攻击的时间间隔是0.5s

3.这个服务是一个集群(这个要求暂时不实现)

编写接口,实现上述需求。

核心问题
可以想到要解决的主要问题是,

1.如何保证一个回合是15秒的时间?

2.如何保证如果一个回合掉落最大金币数量之后,不再掉落金币。

对于问1,我们可以选择设置回合开始的时间或者回合结束的时间,这里采用回合结束的时间。如果发现已经超过结束的时间,那么不做处理。

代码如下,second是一个回合的时间,这里就是十五秒。

private Boolean checkRound(String id, LocalDateTime now) {
    if (Boolean.TRUE.equals(redisTemplate.hasKey(id))) {
        LocalDateTime endTime = (LocalDateTime) redisTemplate.boundValueOps(id).get();
        if (now.isAfter(endTime)) {
            log.info("该回合已经结束:回合id:{}", id);
            return false;
        }
    }
    redisTemplate.boundValueOps(id).set(now.plusSeconds(second));
    return true;
}

对于问2,处理的方式和1一样,redis存储已经掉落的金币,若掉落金币超过最大值,则不予处理。

private Boolean checkMoney(String id) {
    String moneyKey = buildMoneyKey(id);
    if (Boolean.TRUE.equals(stringRedisTemplate.hasKey(moneyKey))) {
        int money = Integer.parseInt(stringRedisTemplate.boundValueOps(moneyKey).get());
        if (money > maxMoney) {
            log.info("金钱超限。回合id:{}", id);
            return false;
        }
    }
    return true;
}

如果当前回合未结束,并且掉落的金币也没有到达最大值,我们将随机生成金币返回去。

private Boolean money(String id){
    Random random = new Random();
    int i = random.nextInt(9);
    if (i <= 2) {
        log.info("获得到了金币:{}", id);
        stringRedisTemplate.boundValueOps(buildMoneyKey(id)).increment();
        return true;
    }
    log.info("未获得到金币:{}", id);
    return false;
}

整体代码逻辑:

@RestController
@Slf4j
public class GameController {
    @Value("${second:15}")
    private Long second;
 
    @Value("${money:2}")
    private Integer maxMoney;
 
    @Resource
    private RedisTemplate redisTemplate;
 
    /**
     * 默认线程池
     */
    @Resource
    private ThreadPoolTaskExecutor threadPoolTaskExecutor;
 
    @Resource
    private StringRedisTemplate stringRedisTemplate;
 
    @GetMapping("/attack")
    public Boolean attack(AttackParam attackParam) {
        String id = attackParam.getRoundId();
        log.info("攻击了一次,回合id:{}", id);
        LocalDateTime now = LocalDateTime.now();
        /**前置检查**/
        if (!preCheck(id, now)) {
            return false;
        }
        return money(id);
    }
 
    /**
     * 检测是否获得金币,获得--true ,未获得--false
     *
     * @param id id
     * @return {@link Boolean}
     */
    private Boolean money(String id){
        Random random = new Random();
        int i = random.nextInt(9);
        if (i <= 2) {
            log.info("获得到了金币:{}", id);
            stringRedisTemplate.boundValueOps(buildMoneyKey(id)).increment();
            return true;
        }
        log.info("未获得到金币:{}", id);
        return false;
    }
 
    private String buildMoneyKey(String id) {
        return "attack:money:" + id;
    }
 
    /**
     * 预检查
     *
     * @param id  id
     * @param now 现在
     * @return {@link Boolean}
     */
    private Boolean preCheck(String id, LocalDateTime now) {
        if (!checkRound(id, now)) {//检查回合
            return false;
        }
        if (!checkMoney(id)) {//检查本回合是否钱已经给够两次了
            return false;
        }
        return true;
    }
 
    /**
     * 校验回合是否结束
     *
     * @param id id
     * @return {@link Boolean}
     */
    private Boolean checkRound(String id, LocalDateTime now) {
        if (Boolean.TRUE.equals(redisTemplate.hasKey(id))) {
            LocalDateTime endTime = (LocalDateTime) redisTemplate.boundValueOps(id).get();
            if (now.isAfter(endTime)) {
                log.info("该回合已经结束:回合id:{}", id);
                return false;
            }
        }
        redisTemplate.boundValueOps(id).set(now.plusSeconds(second));
        return true;
    }
 
    /**
     * 校验金钱是够超限
     *
     * @param id id
     * @return {@link Boolean}
     */
    private Boolean checkMoney(String id) {
        String moneyKey = buildMoneyKey(id);
        if (Boolean.TRUE.equals(stringRedisTemplate.hasKey(moneyKey))) {
            int money = Integer.parseInt(stringRedisTemplate.boundValueOps(moneyKey).get());
            if (money > maxMoney) {
                log.info("金钱超限。回合id:{}", id);
                return false;
            }
        }
        return true;
    }
 
    /**
     * 使用线程池模拟并发测试
     *
     * @return {@link String}
     */
    @GetMapping("/test")
    public String test(){
        AttackParam attackParam = new AttackParam();
        attackParam.setRoundId(UUID.randomUUID().toString());
        for (int i = 0; i <= 10000; i++) {
            CompletableFuture.runAsync(() -> {
                this.attack(attackParam);
            }, threadPoolTaskExecutor);
        }
        return "aa";
    }
}

结果测试
接下来编写代码模拟高并发场景下是否有问题,

本次测试的并发量是1w。

@GetMapping("/test")
public String test(){
    AttackParam attackParam = new AttackParam();
    attackParam.setRoundId(UUID.randomUUID().toString());
    for (int i = 0; i <= 10000; i++) {
        CompletableFuture.runAsync(() -> {
            this.attack(attackParam);
        }, threadPoolTaskExecutor);
    }
    return "aa";
}

测试结束,查询本回合掉落金币数量。

为什么我们设置的最大掉落金币数量是2,结果却是4呢?

好吧,进行第二次测试查看结果。

这一次居然是7。

说明上面这串代码在并发情况下会出现问题,即使这个并发量几十的情况依然会出问题。

问题分析
那我们就来分析一下是哪里出现了问题,出现这种原因无非就是满足写后读,那就找到读写金币的位置。

举个例子,假设线程A正在获取金币,但是这个增加的操作还没有写到redis。另外有线程B,线程C....走到了图二中查询金币数量的位置。那么这一堆线程获得仍是oldValue,这就相当于线程A的写操作是“无效的”。那么导致的结果就是金币比预期多了很多,至于多多少,取决于金币掉落的概率。

解决方案
如何解决这个问题呢?

这个问题本质上是读写分离,导致了“脏数据”。

第一个想到的也是最直接的方法肯定是加锁,但是需要考虑到这种加锁的方式只适合单体应用,如果是多个程序呢,就无法解决了。

可以将synchronized换成分布式锁。

但是加锁的方式不推荐,锁的竞争会严重影响性能。如果可以通过业务逻辑来解决,就不要去加锁。那么我们需要将读写操作放在一起,使其具有原子性。

redis中的incr操作本身就是原子的,所以我们可以将检查金币数量这个操作提前,读写放到一起。

代码如下,checkMoney就可以注掉了。

private Boolean money(String id) {
    Random random = new Random();
    int i = random.nextInt(9);
    if (i <= 2) {
        Long increment = stringRedisTemplate.boundValueOps(buildMoneyKey(id)).increment();//将读和写放到一起 这是个原子性的
        if (increment > maxMoney) {
            log.info("金钱超限,回合{}", id);
            return false;
        }
        log.info("获得到了金币:{}", id);
        stringRedisTemplate.boundValueOps(id+"money").increment();
        return true;
    }
    log.info("未获得到金币:{}", id);
    return false;
}

再次测试,可以看到数据已经是准确的了。
总结
本文讲述了redis在实际业务场景中的应用,并且看到高并发下会产生的数据错误的问题,可采取分布式锁和修改业务逻辑的方式解决,由于锁会影响到性能(请求对锁的竞争),所以更推荐后者。

标签:回合,场景,return,String,redis,private,金币,并发,id
From: https://www.cnblogs.com/chengyiyuki/p/18093088

相关文章

  • JUC并发编程(六)
    1、无锁实现保护共享资源       此前提到过,多个线程同时操作共享资源会导致安全问题,常见的方式是通过加锁(synchronized,reentranlock)解决。但是很显然加锁在某些场景下会影响性能,是否有一种方式可以不用加锁,且保证线程安全?下面来看一个案例1.1、案例一@Slf4j(topic......
  • linux 下安装mysql redis
    查看是否安装mysql:rpm-qa|grepmysql获取mysql版本:wget-i-chttp://dev.mysql.com/get/mysql-community-release-el7-5.noarch.rpm安装:rpm-ivhmysql-community-release-el7-5.noarch.rpmyuminstallmysql-community-serversystemctlstartmysqldsystemctlrest......
  • 在Java项目中使用Redis的五大数据结构应用场景与代码实现
    在Java项目中使用Redis的五大数据结构可以应用于以下场景:1、字符串(String):1、缓存数据:将经常访问的数据存储在Redis中,以减轻数据库的负载。2、计数器:记录用户的访问次数、点赞数等。3、分布式锁:在分布式环境下实现互斥访问,防止并发问题。2、列表(List):1、消息队列:将生产......
  • ngix7种应用场景
           ......
  • MySQL 与 Redis 如何实现最终一致性的四种方案
    背景缓存是软件开发中一个非常有用的概念,数据库缓存更是在项目中必然会遇到的场景。而缓存一致性的保证,针对不同的要求,选择恰到好处的一致性方案。缓存是什么存储的速度是有区别的。缓存就是把低速存储的结果,临时保存在高速存储的技术。如图所示,金字塔上层的存储,可以作为下......
  • 解决长尾问题,BEV-CLIP:自动驾驶中复杂场景的多模态BEV检索方法
    解决长尾问题,BEV-CLIP:自动驾驶中复杂场景的多模态BEV检索方法理想汽车的工作,原文,BEV-CLIP:Multi-modalBEVRetrievalMethodologyforComplexSceneinAutonomousDriving链接:https://arxiv.org/pdf/2401.01065.pdf自动驾驶中对复杂场景数据的检索需求正在增加,尤其是随着......
  • @Transactional注解失效场景以及解决方法
    该文章专注于面试,面试只要回答关键点即可,不需要对框架有非常深入的回答,如果你想应付面试,是足够了,抓住关键点面试官:说一说@Transactional注解失效的场景以及解决方法@Transactional是Spring框架提供的一个注解,用于声明事务的边界。它可以应用于类、方法或接口上,用于指定......
  • Golang: 通过chan来实现并发访问控制
    通过chan来实现并发访问控制通过chan来实现并发访问控制背景介绍这是在阅读grom的源码时,他的schema的初始化方式,给我留下来很深刻的印象,本文将通过channel的一些使用来实现实例的并发访问技术要点如果chan为空时,尝试读可以成功,获得的结果为空示例代码packagemai......
  • etcd 以及 redis分布式锁的实现优劣比较
    etcd以及redis分布式锁的实现优劣比较背景介绍在学习etcd时,对于使用etcd实现分布式锁(使用etcd来实现一个简单的分布式锁)做了一个简单的示例,同时也能想到和Redis实现的分布式锁相比,基于etcd来做有什么好处呢?技术要点底层技术比较我们必须要明白一件事情,两者的底......
  • Golang: 通过chan来实现并发访问控制
    通过chan来实现并发访问控制通过chan来实现并发访问控制背景介绍这是在阅读grom的源码时,他的schema的初始化方式,给我留下来很深刻的印象,本文将通过channel的一些使用来实现实例的并发访问技术要点如果chan为空时,尝试读可以成功,获得的结果为空示例代码packagemai......