如何使用Redis和RabbitMQ实现一个学生抢课系统(可类比商品秒杀系统)
1、如何使用Redis和RabbitMQ实现一个学生抢课系统(可类比商品秒杀系统)
- 电商项目中的秒杀场景我们都很常见,不只是京东和淘宝现在很多的小程序公众号也有做现时限购的秒杀场景,那么如何做一个秒杀系统呢?
- 把数据全部放在MySQL数据中,这显然是不现实的,因为在秒杀开始时会有大量的请求涌进来,如果直接访问数据库,我们的MySQL很容易崩点,我们就应该做一层过滤。
如下就是实现一个抢课系统的来类比一个秒杀系统。
我们这里引入Redisson,介绍如下:
- Redisson是一个在Redis的基础上实现的Java驻内存数据网格(In-Memory Data Grid)。它不仅提供了一系列的分布式的Java常用对象,还提供了许多分布式服务。其中包括(BitSet, Set, Multimap, SortedSet, Map, List, Queue, BlockingQueue, Deque, BlockingDeque, Semaphore, Lock, AtomicLong, CountDownLatch, Publish / Subscribe, Bloom filter, Remote service, Spring cache, Executor service, Live Object service, Scheduler service) Redisson提供了使用Redis的最简单和最便捷的方法。Redisson的宗旨是促进使用者对Redis的关注分离(Separation of Concern),从而让使用者能够将精力更集中地放在处理业务逻辑上。
而在我们的这个秒杀程序中主要运用了Redisson中的信号量(Semaphore)。
在抢课之前我们先将课程进行一个上架操作,主要就是将课程的最高选课人数作为信号量存入redis中。关键代码如下:
@Autowired
private CourseMapper courseMapper;
@Autowired
private RedissonClient redissonClient;
private final String COURSE_SEMAPHORE = "subject:course:num:";//
@Test
public void contextLoads() {
log.info("开始上架课程");
List<Course> sessions = courseMapper.selectOkCourse();
sessions.stream().forEach(session ->{
String jsonString = JSON.toJSONString(session);
//引入分布式的信号量 用信号量控制并发数,限流
RSemaphore semaphore = redissonClient.getSemaphore(COURSE_SEMAPHORE + session.getId());
//课程的最大人数作为信号量
semaphore.trySetPermits(session.getNum());
});
}
在学生抢课的过程中,我们会做一下几步操作:
1、校验该学生是否已经讲过该课程
2、校验该课程是否已经达到最大选课人数
3、该学生的选课信息放入倒RabbitMQ中,由其他模块进行异步处理学生选课入库
代码如下:
public String subject(String studentId, String courseId) {
log.info("抢课任务开始{},{}",studentId,courseId);
Long ttl = 24*3600L; //过期时间根据具体情况配置,这里为了简便写了两天
//2.4验证学生是否已经抢过该课(幂等性)
//setnx不存在的时候才占位
String redisKey = studentId+"_"+courseId;
Boolean aBoolean = redisTemplate.opsForValue().setIfAbsent(redisKey, "1".toString(), ttl, TimeUnit.SECONDS);
if(aBoolean) {
//成功表示该学生第一次选课
//3.获取信号量tryAcquire()非阻塞方法
RSemaphore semaphore = redissonClient.getSemaphore(COURSE_SEMAPHORE + courseId);
//获取信号量,同一个课程,同一个学生只能选一次,故这里获取一个信号量,若是商品可根据限购几件来设定
boolean b = semaphore.tryAcquire(1);
if(b){
//还有信号量,抢课信息入库 MQ登场
log.info("抢课完成");
SubjectTo subjectTo = new SubjectTo(studentId,courseId);
rabbitTemplate.convertAndSend("course-event-exchange","course.seckill.course",subjectTo);
log.info("发送完成");
return "1";
}else {
//该课程已经被抢完
log.error("该课程已经被抢完");
return "3";
}
}else{
//已经选过该课程
log.error("该过程已经抢过");
return "2";
}
}
监听MQ的入库程序如下:
@RabbitListener(queues = "seckill.course.queue")
public void handleSecKillOrderCreate(SubjectTo to) {
try {
log.info("---------抢课信息准备入库-----------");
//加入课程学生表中
courseService.makeItCoures(to);
//该课程剩余可选人数减少1(主要是为了展示课程剩余人数来使用)
courseService.changeNowNum(to.getCourseId());
} catch (Exception e) {
e.printStackTrace();
}
}
以上就上学生抢课系统的主要思路,比较简介。其中步骤主要分为如下:
1、课程上架
2、学生抢课模块,抢课校验,课程信息放MQ中
3、监听MQ中的选课信息,进行入库操作。
个人公众号 :hellotqq,坚持原创输出,坚持分享学习感悟,下方扫码关注,与您共同成长!

如何使用Redis和RabbitMQ实现一个学生抢课系统(可类比商品秒杀系统)的更多相关文章
- 用Redis轻松实现秒杀系统
秒杀系统的架构设计 秒杀系统,是典型的短时大量突发访问类问题.对这类问题,有三种优化性能的思路: 写入内存而不是写入硬盘 异步处理而不是同步处理 分布式处理 用上这三招,不论秒杀时负载多大,都能轻松应 ...
- 缓存、队列(Memcached、redis、RabbitMQ)
本章内容: Memcached 简介.安装.使用 Python 操作 Memcached 天生支持集群 redis 简介.安装.使用.实例 Python 操作 Redis String.Hash.Li ...
- 缓存、队列(Memcached,Redis,rabbitMQ)
一.Memcached Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站的 ...
- Python之路【第十篇】Python操作Memcache、Redis、RabbitMQ、SQLAlchemy、
Memcached Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站的速度 ...
- 缓存大全(Memcached、redis、RabbitMQ )
Memcached: 简介.安装.使用 python操作Memcached Memcached天生支持集群 Redis: 简介.安装.使用.实例 Python操作Redis String.Hash.L ...
- Python-操作Memcache、Redis、RabbitMQ、
Memcache 简述: Memcache是一套分布式的高速缓存系统,由LiveJournal的Brad Fitzpatrick开发,但目前被许多网站使用以提升网站的访问速度,尤其对于一些大型的.需要 ...
- Python自动化运维之17、Python操作 Memcache、Redis、RabbitMQ
一.Memcached Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站的 ...
- Python操作 Memcache、Redis、RabbitMQ、SQLAlchemy
Memcached Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站的速度 ...
- centos7.6编译安装php7.2.11及redis/memcached/rabbitmq/openssl/curl等常见扩展
centos7.6编译安装php7..11及redis/memcached/rabbitmq/openssl/curl等常见扩展 获取Php的编译参数方法: [root@eus-api-cms-bac ...
随机推荐
- Java学习之路:流程控制
2022-10-11 10:58:41 前言 本文开始流程控制方面的学习,主要包括用户交互和流程控制语句,适合新手学习. 1 用户交互Scanner 1.1 Scanner对象 Java提供了一个可以 ...
- Maven中POM文件总体配置说明
POM文件总体配置说明 <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://w ...
- Oracle注入
Oracle 查询出所有的表 select * from all_tables 查询出当前用户的表 select * from user_tables 查询出所有的字段 select*from all ...
- 8.RabbitMQ系列之RPC
1. RPC Remote Procedure Call:远程过程调用,一次远程过程调用的流程即客户端发送一个请求到服务端,服务端根据请求信息进行处理后返回响应信息,客户端收到响应信息后结束 2. C ...
- .Net Core&RabbitMQ限制循环消费
前言 当消费者端接收消息处理业务时,如果出现异常或是拒收消息将消息又变更为等待投递再次推送给消费者,这样一来,则形成循环的条件. 循环场景 生产者发送100条消息到RabbitMQ中,消费者设定读取到 ...
- 我用EasyExcel优化了公司的导出(附踩坑记录)
背景介绍 最近要改一个导出的功能,在原有的基础上,在导出一份明细数据,要求导出内容加在原有 excel 的第二个 sheet 上.考虑到数据量还比较大,干脆引入阿里的 EasyExcel 来做. 下面 ...
- 在 .NET 7上使用 WASM 和 WASI
WebAssembly(WASM)和WebAssembly System Interface(WASI)为开发人员开辟了新的世界..NET 开发人员在 Blazor WebAssembly 发布时熟悉 ...
- Seata Server 1.5.2 源码学习
Seata 包括 Server端和Client端.Seata中有三种角色:TC.TM.RM,其中,Server端就是TC,TM和RM属Client端.Client端的源码学习上一篇已讲过,详见 < ...
- js高级基础部分
基于尚硅谷的尚硅谷JavaScript高级教程提供笔记撰写,加入一些个人理解 github源码 博客下载 数据类型的分类和判断 主要问题 分类 基本(值)类型 Number ----- 任意数值 -- ...
- Go map 竟然也会发生内存泄露?
Go 程序运行时,有些场景下会导致进程进入某个"高点",然后就再也下不来了. 比如,多年前曹大写过的一篇文章讲过,在做活动时线上涌入的大流量把 goroutine 数抬升了不少,流 ...