SpringBoot开发案例之整合ActiveMQ实现秒杀队列

前言

在实际生产环境中中,通常生产者和消费者会是两个独立的应用,这样才能通过消息队列实现了服务解耦和广播。因为此项目仅是一个案例,为了方便期间,生产和消费定义在了同一个项目中。

安装说明

参考:Docker学习之搭建ActiveMQ消息服务

基础配置

pom.xml 添加依赖:

<!-- activemq -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-activemq</artifactId>
</dependency>

application.properties 基础配置:

# activemq 基础配置
#spring.activemq.broker-url=tcp://47.94.232.109:61616
# 生产环境设置密码
#spring.activemq.user=admin
#spring.activemq.password=123456
#spring.activemq.in-memory=true
#spring.activemq.pool.enabled=false

项目集成

定义生产者:

import javax.jms.Destination;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jms.core.JmsMessagingTemplate;
import org.springframework.stereotype.Component;

@Component
public class ActiveMQSender {
    
    @Autowired
    private JmsMessagingTemplate jmsTemplate;
    
    /*
     * 发送消息,destination是发送到的队列,message是待发送的消息
     */
    public void sendChannelMess(Destination destination, final String message){
        jmsTemplate.convertAndSend(destination, message);
    }

}

定义消费者:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jms.annotation.JmsListener;
import org.springframework.stereotype.Service;

import com.itstyle.seckill.common.entity.Result;
import com.itstyle.seckill.common.enums.SeckillStatEnum;
import com.itstyle.seckill.common.redis.RedisUtil;
import com.itstyle.seckill.common.webSocket.WebSocketServer;
import com.itstyle.seckill.service.ISeckillService;

@Service
public class ActiveMQConsumer {
    
    @Autowired
    private ISeckillService seckillService;
    @Autowired
    private RedisUtil redisUtil;
    
    // 使用JmsListener配置消费者监听的队列,其中text是接收到的消息
    @JmsListener(destination = "seckill.queue")
    public void receiveQueue(String message) {
        //收到通道的消息之后执行秒杀操作(超卖)
        String[] array = message.split(";"); 
        Result result = seckillService.startSeckilDBPCC_TWO(Long.parseLong(array[0]), Long.parseLong(array[1]));
        if(result.equals(Result.ok(SeckillStatEnum.SUCCESS))){
            WebSocketServer.sendInfo(array[0].toString(), "秒杀成功");//推送给前台
        }else{
            WebSocketServer.sendInfo(array[0].toString(), "秒杀失败");//推送给前台
            redisUtil.cacheValue(array[0], "ok");//秒杀结束
        }
    }
}

测试案例:

@ApiOperation(value="秒杀五(ActiveMQ分布式队列)",nickname="科帮网")
@PostMapping("/startActiveMQQueue")
public Result startActiveMQQueue(long seckillId){
    seckillService.deleteSeckill(seckillId);
    final long killId =  seckillId;
    LOGGER.info("开始秒杀五");
    for(int i=0;i<1000;i++){
        final long userId = i;
        Runnable task = new Runnable() {
            @Override
                public void run() {
                if(redisUtil.getValue(killId+"")==null){
                    Destination destination = new ActiveMQQueue("seckill.queue");
                    //思考如何返回给用户信息ws
                    activeMQSender.sendChannelMess(destination,killId+";"+userId);
                }else{
                    //秒杀结束
                }
            }
        };
        executor.execute(task);
    }
    try {
        Thread.sleep(10000);
        redisUtil.cacheValue(killId+"", null);
        Long  seckillCount = seckillService.getSeckillCount(seckillId);
        LOGGER.info("一共秒杀出{}件商品",seckillCount);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
    return Result.ok();
}

注意事项

  • spring-boot-starter-activemq 依赖即可默认采用内嵌的 ActiveMQ,这个跟 elasticsearch 是一样的,测试的小伙伴可以不用安装,注释掉相关参数,使用默认即可。
  • 如果自行安装 ActiveMQ 记得配置防火墙/安全组,配置web访问密码以及连接密码。
  • 在生产环境下尽量还是采用外部 activemq 服务,提高扩展性、稳定性、可维护性。

参考

spring-boot-starters

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏耕耘实录

RHEL7.X系列及周边Linux发行版中,关于MBR与GPT的选择一些思考与建议

存储的选型、规划与管理等工作一直以来都是日常系统运维工作中的重点。MBR与GPT两种类型的分区表的选择与使用则是在磁盘管理中需要根据应用场景来注或考虑的要点。结...

13920
来自专栏乐沙弥的世界

基于Linux (RHEL 5.5) 安装Oracle 10g RAC

    本文所描述的是在Red Hat 5.5下使用vmware server 来安装Oracle 10g RAC(OCFS + ASM),本文假定你的RHEL...

17130
来自专栏Java帮帮-微信公众号-技术文章全总结

Web-第十三天 基础加强-JDBC高级开发事务【悟空教程】

insert into account values (null,'jack',10000);

9720
来自专栏吴生的专栏

使用Spring AOP实现MySQL数据库读写分离案例分析

分布式环境下数据库的读写分离策略是解决数据库读写性能瓶颈的一个关键解决方案,更是最大限度了提高了应用中读取 (Read)数据的速度和并发量。

16820
来自专栏java、Spring、技术分享

java 日志处理

  common-logging是 apache提供的一个通用的日志接口。用户可以自由选择第三方的日志组件作为具体实现,像log4j,或者jdk自带的loggi...

47830
来自专栏乐沙弥的世界

Linux 6 下编译安装 PHP 5.6

25720
来自专栏Flutter&Dart

DartVM服务器开发(第十九天)--jaguar_reflect使用Controller

上面有一个ReflectedController(UserController()).routes,就是把UserController里面的接口反射出来,添加到...

15430
来自专栏java工会

推荐几个自己写的Java后端相关的范例项目

26850
来自专栏Ryan Miao

AngularJS中使用service,并同步数据

service是单例对象,在应用中不同代码块之间共享数据。 对一些公用的方法封装到service中,然后通过依赖注入在Controller中调用,示例代码: 1...

392100
来自专栏Java Web

MyBatis 与 Spring 整合

MyBatis—Spring 项目 目前大部分的 Java 互联网项目,都是用 Spring MVC + Spring + MyBatis 搭建平台的。 使用 ...

45160

扫码关注云+社区

领取腾讯云代金券