首页 > 其他分享 >校园课程助手【8】-RabbitMQ实现异步选课

校园课程助手【8】-RabbitMQ实现异步选课

时间:2024-08-06 18:25:10浏览次数:25  
标签:异步 return 选课 RabbitMQ public getId order CourseVo user

本节是此项目核心问题,保证在高并发情况下选课业务能够高效、正确的完成。

1.在进行选课前将课程库存提前加载到Redis中:

//在抢课Controller中实现InitializingBean接口

    //初始化时执行将库存预加载到Redis
    @Override
    public void afterPropertiesSet() throws Exception {
        List<CourseVo> list = courseService.findCourseVo();
        if (CollectionUtils.isEmpty(list)) {
            return;
        }
        list.forEach(CourseVo-> {
            redisTemplate.opsForValue().set("seckillCourese:"+CourseVo.getId(),CourseVo.getStockCount());
            if(CourseVo.getStockCount() > 0)
                emptyStockMap.put(CourseVo.getId(),false);
            else
                emptyStockMap.put(CourseVo.getId(),true);
        });
    }

2.抢课Controller

//使用前后端分离,对象缓存减少前端页面的数据访问,同时使用Redis判断是否重复抢购
@RequestMapping(value = "/{path}/doSeckill",method = RequestMethod.POST)
@ResponseBody
public RespBean doSecKill(@PathVariable String path,User user,Long CourseId){
        if(user == null)
        return RespBean.error(RespBeanEnum.SESSION_ERROR);

        boolean check = orderService.checkPath(user,CourseId,path);
        if(!check){
        return RespBean.error(RespBeanEnum.REQUEST_ILLEGAL);
        }
        //判断是否重复抢购
        TSeckillOrder seckillOrder =
        (TSeckillOrder) redisTemplate.opsForValue().get("order:"+user.getId()+":"+CourseId);
        if(seckillOrder != null){
        return RespBean.error(RespBeanEnum.REPEATE_ERROR);
        }
        if(emptyStockMap.get(CourseId)){
        return  RespBean.error(RespBeanEnum.EMPTY_STOCK);
        }
        //通过Redis预减库存
        ValueOperations valueOperations = redisTemplate.opsForValue();
        //原子性预减库存操作
        Long stock = valueOperations.decrement("seckillCourse:"+CourseId);

        //也可以使用Redis结合lua脚本
//        Long stock = (Long) redisTemplate.execute(script, Collections.singletonList("seckillCourse:"+CourseId),
//                Collections.EMPTY_LIST);
        if(stock < 0){
        emptyStockMap.put(CourseId,true);
        valueOperations.increment("seckillCourse:"+CourseId);
        return RespBean.error(RespBeanEnum.EMPTY_STOCK);
        }
        SeckillMessage seckillMessage = new SeckillMessage(user, CourseId);
        mqSender.sendSeckillMessage(JsonUtil.object2JsonStr(seckillMessage));
//        通过RabbitMQ消息队列下单,0状态表示排队中
        return  RespBean.success(0);
        }

3.配置RabbitMQ


@Configuration
public class RabbitMQTopicConfig {

    private static final String QUEUE = "seckillQueue";
    private static final String EXCHANGE = "seckillExchange";

    @Bean
    public Queue queue() {
        return new Queue(QUEUE);
    }

    @Bean
    public TopicExchange topicExchange() {
        return new TopicExchange(EXCHANGE);
    }

    @Bean
    public Binding binding() {
        return BindingBuilder.bind(queue()).to(topicExchange()).with("seckill.#");
    }

4.消息发送


@Service
@Slf4j
public class MQSender {
    @Autowired
    private RabbitTemplate rabbitTemplate;

    //发送秒杀信息
    public void sendSeckillMessage(String msg){
        log.info("发送"+msg);
        rabbitTemplate.convertAndSend("seckillExchange","seckill.msg",msg);
    }
}

5.消息接收


@Service
@Slf4j
public class MQReceiver {

    @Autowired
    private ITCourseService courseService;
    @Autowired
    private RedisTemplate redisTemplate;
    @Autowired
    private ITOrderService orderService;

    //接收消息,实际上进行下单操作
    @RabbitListener(queues = "seckillQueue")
    public void receive(String msg){
        log.info("接收"+msg);
        SeckillMessage seckillMessage = JsonUtil.jsonStr2Object(msg, SeckillMessage.class);
        Long courseId = seckillMessage.getCourseId();
        User user = seckillMessage.getUser();
        CourseVo courseVobyCourseId= courseService.findCourseVobyCourseId(courseId );
        if(courseVobyCourseId.getStockCount() < 1){
            return;
        }
        //判断是否重复抢购
        TSeckillOrder seckillOrder =
                (TSeckillOrder) redisTemplate.opsForValue().get("order:"+user.getId()+":"+courseId );
        if(seckillOrder != null){
            return ;
        }
        //下单
        orderService.secKill(user,courseVobyCourseId);
    }
}

6.选课成功sevice下单逻辑


@Transactional
@Override
public TOrder secKill(User user, CourseVo CourseVo) {
        ValueOperations valueOperations = redisTemplate.opsForValue();
        //从后端重新查询库存,拿到秒殺商品信息
        TSeckillCourse seckillCourse = itSeckillCourseService.getOne(new QueryWrapper<TSeckillCourse>().eq("Course_id", CourseVo.getId()));
        //库存减一
        seckillCourse.setStockCount(seckillCourse.getStockCount() - 1);
        //id没问题同时库存>0才更新
        boolean seckillCourseResult = itSeckillCourseService.update(new UpdateWrapper<TSeckillCourse>()
        .setSql("stock_count = " + "stock_count-1")
        .eq("Course_id", CourseVo.getId())
        .gt("stock_count", 0)
        );
        if (seckillCourse.getStockCount() < 1) {
        valueOperations.set("isStockEmapty:"+ CourseVo.getId(),"0");
        return null;
        }

        //生成订单
        TOrder order = new TOrder();
        order.setUserId(user.getId());
        order.setCourseId(CourseVo.getId());
        order.setDeliveryAddrId(0L);
        order.setCourseName(CourseVo.getCourseName());
        order.setCourseCount(1);
        order.setCoursePrice(seckillCourse.getSeckillPrice());
        order.setOrderChannel(1);
        order.setStatus(0);
        order.setCreateDate(new Date());
        tOrderMapper.insert(order);
        TSeckillOrder tSeckillOrder = new TSeckillOrder();
        tSeckillOrder.setUserId(user.getId());
        tSeckillOrder.setOrderId(order.getId());
        tSeckillOrder.setCourseId(CourseVo.getId());
        itSeckillOrderService.save(tSeckillOrder);
        
        //这里使用Redis缓存用户id和订单id作为联合key,在高并发时订单控制器可以先进行查询防止一个人多次
        redisTemplate.opsForValue().set("order:" + user.getId() + ":" + CourseVo.getId(),
        tSeckillOrder, 1, TimeUnit.MINUTES);
        return order;
        }

7.前端通过轮询得知抢课是否成功

function doSeckill(path) {
        $.ajax({
            url: '/seckill/'+path+'/doSeckill',
            type: "POST",
            data: {
                courseId: $('#courseId').val()
                // path:path改用注解接收
            },
            success: function (data) {
                if (data.code == 200) 
                    //使用RabbitMQ轮询时
                    getResult($("#courseId").val());
                } else {
                    layer.msg(data.message);
                }
            }, error: function () {
                layer.msg("客户端请求出错");
            }
        });
    }
    function getResult(courseId){
        g_showLoading();
        $.ajax({
            url:"/seckill/result",
            type:"GET",
            data: {
                courseId: courseId
            },
            success: function (data) {
                if (data.code == 200) {
                    var result = data.object;
                    if (result < 0) {
                        layer.msg("对不起,秒杀失败");
                    } else if (result == 0) {
                        setTimeout(function () {
                            getResult(courseId)
                        },50);
                    } else {
                        layer.confirm("恭喜您,秒杀成功!查看订单?", {btn: ["确定", "取消"]},
                            function () {
                                window.location.href = "/orderDetail.htm?orderId=" + result;
                            },
                            function () {
                                layer.close();
                            }
                        )
                    }
                }
            },
            error: function () {
                layer.msg("客户端请求错误");
            }
        });
    }
    //获取抢课结果,成功返回订单id,失败-1,正在排队0
    @RequestMapping(value = "/result",method = RequestMethod.GET)
    @ResponseBody
    public RespBean getResult(User user, Long goodsId){
        if(user == null){
            return  RespBean.error(RespBeanEnum.SESSION_ERROR);
        }
        Long orderId = seckillOrderService.getResult(user,goodsId);
        return  RespBean.success(orderId);
    }

标签:异步,return,选课,RabbitMQ,public,getId,order,CourseVo,user
From: https://blog.csdn.net/qq_43954910/article/details/140921262

相关文章

  • 串行通信协议--UART(Universal Asynchronous Receiver/Transmitter,通用异步收发传输器
    一、UART简介  UART广泛应用于微控制器和计算机之间的数据通信,如GPS模块、蓝牙模块、GSM模块等。UART是一种通用串行数据总线,用于异步通信,该总线双向通信,可以实现全双工传输和接收。在嵌入式设计中,UART用于主机与辅助设备通信UART通常被集成于其他通讯接口的连结上。UA......
  • SSM高校学生在线选课系统q399g 系统界面在最后面
    本系统(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。系统程序文件列表系统内容:教师,学生,课程信息,班级信息,选课信息,打分信息,学分信息,必修课程,公告信息开题报告内容课题名称:SSM高校学生在线选课系统一、研究背景与意义随着......
  • 异步编程和多线程有
    在C#中,多线程和异步编程是两个相关但不完全相同的概念。下面我会解释这两个概念的区别,并给出一些常见的问题及解答。多线程vs异步编程多线程:多线程指的是在一个进程中创建多个线程来并行执行任务。多线程可以用来处理计算密集型任务,充分利用多核处理器的计算能力。多......
  • 【Rabbitmq的消息模型】
    消息队列的特性durable:队列持久化。如果设置持久化,那么无论RabbitMQ在关闭时,就会将队列存储到本地磁盘,无论宕机还是重启,队列也不会删除;如果设置不持久化,那么在RabbitMQ关闭时,就会将队列删除。exclusive:独占队列。如果设置独占,那么当前队列只允许预先设置的Connection访问;如......
  • windows下RabbitMQ安装后,无法进入web管理页面问题
    在window安装rabbitmq,访问http://127.0.0.1:15672/,访问不了;有可能是没开启网页管理界面1、在cmd窗口下进入rabbitmq安装目录下的sbin目录,使用rabbitmq-plugins.batlist查看已安装的插件列表。 2、使用rabbitmq-plugins.batenablerabbitmq_management开启网页管理界面 ......
  • 基于SpringBoot+Vue+uniapp的学生网上选课系统的详细设计和实现(源码+lw+部署文档+讲
    文章目录前言详细视频演示具体实现截图技术栈后端框架SpringBoot前端框架Vue持久层框架MyBaitsPlus系统测试系统测试目的系统功能测试系统测试结论为什么选择我代码参考数据库参考源码获取前言......
  • RabbitMQ(三)Java客户端
    1.快速入门在idea里面创建两个springboot项目,一个模块是consumer,一个是publisher两者有自己的启动类,继承同一父工程的pom。父工程的pom.xml<?xmlversion="1.0"encoding="UTF-8"?><projectxmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http......
  • 如何理解js的异步
    js是一门单线程的语言,这是因为他运行在浏览器的渲染主线程中,而渲染主线程只有一个,渲染主线程担任着诸多的工作,渲染页面、执行js、css、计时器等等都在其中运行。如果使用同步的方式,就很有可能会导致主线程堵塞,从而导致消息队列中的其它任务无法进行执行,这样一来,一方面会导致......
  • 计算机毕业设计必看必学!! 85583 springboot高校网上选课系统,原创定制程序, java、PHP
                                                  摘要本论文主要论述了如何使用JAVA语言开发一个高校网上选课系统,本系统将严格按照软件开发流程进行各个阶段的工作,采用B/S架构,面向对象编程思想进行项目开发。在引言中,......
  • 记一次JSF异步调用引起的接口可用率降低
    前言本文记录了由于JSF异步调用超时引起的接口可用率降低问题的排查过程,主要介绍了排查思路和JSF异步调用的流程,希望可以帮助大家了解JSF的异步调用原理以及提供一些问题排查思路。本文分析的JSF源码是基于JSF1,7.5-HOTFIX-T6版本。起因问题背景1.广告投放系统是典型的I/O密......