文章目录

第 6 章 消息中间件RabbitMQ

学习目标:1

理解 RabbitMQ 的主要概念,完成 RabbitMQ 的安装
理解直接模式和分列模式,说出直接模式和分列模式在电商系统的应用场景
完成用户注册,能够将消息发送给 RabbitMQ
完成短信服务,能够接收消息并调用阿里云通信完成短信发送

1. 走进 RabbitMQ

1.1 消息中间件简介 41

一个系统向另外一个系统发送消息, 并非点对点关系, 否则需要维护大量的链接
使用中间件, 吸收转化消息, 可以简化流程和逻辑, 并因此有其他优势
比如异步, 双方不必实时接受消息, dubbo 为同步, 双方必须在线, 如果一方处理耗时很长, 将严重降低整体性能
比如解耦, 直接调用不如间接调用, 调用模块不必指名道姓, 而是将需求交给中间件, 中间件调用, 隐藏后台
比如流量削锋, 如果存在大量消息并发, 比如发短信, 不可直接调用发短信接口, 可以将所有消息存储到中间件, 然后另外一个进程再从中间件中提取即可,

消息中间件(消息队列)是分布式系统中重要的组件,主要解决应用耦合,异步消息,流量削锋等问题实现高性能,高可用,可伸缩和最终一致性 [架构] 使用较多的消息队列有 ActiveMQ,RabbitMQ,ZeroMQ,Kafka,MetaMQ,RocketMQ
以下介绍消息队列在实际应用中常用的使用场景:异步处理,应用解耦,流量削锋和消息通讯四个场景
kafka 效率高, 但是有几率数据丢失, 一般大数据时使用

1.2 什么是 RabbitMQ

RabbitMQ 是一个由 Erlang 语言开发的 AMQP 的开源实现。
AMQP :Advanced Message Queue,高级消息队列协议。它是应用层协议的一个开放标准,为面向消息的中间件设计,基于此协议的客户端与消息中间件可传递消息,并不受产品、开发语言等条件的限制。
消息中间件有很多协议
RabbitMQ 最初起源于金融系统,用于在分布式系统中存储转发消息,在易用性、扩展性、高可用性等方面表现不俗。具体特点包括:

  1. 可靠性(Reliability)
    RabbitMQ 使用一些机制来保证可靠性,如持久化 (消息存到磁盘中防丢失)、传输确认、发布确认。
  2. 灵活的路由(Flexible Routing)
    配置转发
    在消息进入队列之前,通过 Exchange 来路由消息的。对于典型的路由功能,RabbitMQ 已经提供了一些内置的 Exchange 来实现。针对更复杂的路由功能,可以将多个 Exchange 绑定在一起,也通过插件机制实现自己的 Exchange 。
  3. 消息集群(Clustering)
    高并发场景
    多个 RabbitMQ 服务器可以组成一个集群,形成一个逻辑 Broker 。
  4. 高可用(Highly Available Queues)
    队列可以在集群中的机器上进行镜像,使得在部分节点出问题的情况下队列仍然可用。
  5. 多种协议(Multi-protocol)
    RabbitMQ 支持多种消息队列协议,比如 STOMP、MQTT 等等。
  6. 多语言客户端(Many Clients)
    RabbitMQ 几乎支持所有常用语言,比如 Java、.NET、Ruby 等等。
  7. 管理界面(Management UI)
    RabbitMQ 提供了一个易用的用户界面,使得用户可以监控和管理消息 Broker 的许多方面。
  8. 跟踪机制(Tracing)
    如果消息异常,RabbitMQ 提供了消息跟踪机制,使用者可以找出发生了什么。
  9. 插件机制(Plugin System)
    RabbitMQ 提供了许多插件,来从多方面进行扩展,也可以编写自己的插件。

1.3 架构图与主要概念

1.3.1 架构图

消息生产者, 交换器, 队列, 消费者,
队列是消息存储模块, 交换器控制消息转发规则, 将消息发送到对应的队列
就像快递员负责发快递, 控制快递走向, 而快递点负责存储

交换器使用路由 key 指定绑定规则, 可以是一个队列名等等

1.3.2 主要概念

RabbitMQ Server: 也叫 broker server,它是一种传输服务。 他的角色就是维护一条从 Producer 到 Consumer 的路线,保证数据能够按照指定的方式进行传输。
Producer: 消息生产者,如图 A、B、C,数据的发送方。消息生产者连接 RabbitMQ 服务器然后将消息投递到 Exchange。
Consumer:消息消费者,如图 1、2、3,数据的接收方。消息消费者订阅队列,
RabbitMQ 将 Queue 中的消息发送到消息消费者。
Exchange:生产者将消息发送到 Exchange(交换器),由 Exchange 将消息路由到一个或多个 Queue 中(或者丢弃)。Exchange 并不存储消息。RabbitMQ 中的 Exchange 有 direct、fanout、topic、headers 四种类型,每种类型对应不同的路由规则。
Queue:(队列)是 RabbitMQ 的内部对象,用于存储消息。消息消费者就是通过订阅队列来获取消息的,RabbitMQ 中的消息都只能存储在 Queue 中,生产者生产消息并最终投递到 Queue 中,消费者可以从 Queue 中获取消息并消费。多个消费者可以订阅同一个 Queue,这时 Queue 中的消息会被平均分摊给多个消费者进行处理,而不是每个消费者都收到所有的消息并处理。
RoutingKey:生产者在将消息发送给 Exchange 的时候,一般会指定一个 routing key,来指定这个消息的路由规则,而这个 routing key 需要与 Exchange Type 及 binding key 联合使用才能最终生效。在 Exchange Type 与 binding key 固定的情况下(在正常使用时一般这些内容都是固定配置好的),我们的生产者就可以在发送消息给 Exchange 时,通过指定 routing key 来决定消息流向哪里。RabbitMQ 为 routing key 设定的长度限制为 255bytes。
Connection: (连接):Producer 和 Consumer 都是通过 TCP 连接到 RabbitMQ Server 的。以后我们可以看到,程序的起始处就是建立这个 TCP 连接。
Channels: (信道):它建立在上述的 TCP 连接中。数据流动都是在 Channel 中进行的。也就是说,一般情况是程序起始建立 TCP 连接,第二步就是建立这个 Channel。
VirtualHost:权限控制的基本单位,一个 VirtualHost 里面有若干 Exchange 和 MessageQueue,以及指定被哪些 user 使用

1.4 RabbitMQ 安装与启动

(1)下载并安装 Eralng 软件 otp_win64_20.2.exe, 提供环境
(2)下载并安装 rabbitmq 软件 rabbitmq-server-3.7.4.exe。双击安装,注意不要安装在包含中文和空格的目录下!安装后 window 服务中就存在 rabbitMQ 了,并且是启动状态。
(3)安装管理界面(插件), 使得服务可视化
进入 rabbitMQ 安装目录的 sbin 目录,输入命令
进入此目录的 cmd
小技巧, 选中目录路径, 输入 cmd 回车即可进入此目录的 cmd

rabbitmq‐plugins enable rabbitmq_management


(4)重新启动服务

(5)打开浏览器,地址栏输入 http://127.0.0.1:15672 , 即可看到管理界面的登陆页

输入用户名和密码,都为 guest 进入主界面:

最上侧的导航以此是:概览、连接、信道、交换器、队列、用户管理
每个用户增删改查以及权限设计

2. RabbitMQ 发送与接收消息

提供了四种模式, 这里说最常见的两种

2.1 直接模式(Direct)

2.1.1 什么是 Direct 模式

我们需要将消息发给唯一一个节点时使用这种模式,这是最简单的一种形式。

任何发送到 Direct Exchange 的消息都会被转发到 RouteKey 中指定的 Queue。

  1. 一般情况可以使用 rabbitMQ 自带的 Exchange:”“(该 Exchange 的名字为空字符串,下文称其为 default Exchange)。
  2. 这种模式下可以不需要将 Exchange 进行任何绑定 (binding) 操作, 其他队列都需要和交换器绑定
  3. 消息传递时需要一个 “RouteKey”,可以简单的理解为要发送到的队列名字。
  4. 如果 vhost 中不存在 RouteKey 中指定的队列名,则该消息会被抛弃。

2.1.2 创建队列

创建队列,名为 queue.test

2.1.3 代码实现 - 消息生产者

新建一个工程
(1)创建工程 rabbitmq_demo,引入依赖 ,pom.xml 如下:

<dependency>
<groupId>org.springframework.amqp</groupId>
<artifactId>spring‐rabbit</artifactId>
<version>2.1.4.RELEASE</version>
</dependency>
 
 

(2)编写配置文件 applicationContext-rabbitmq-producer.xml
这里是生产者, 后面还有消费则, 两者有区别

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	   xmlns:rabbit="http://www.springframework.org/schema/rabbit"
	   xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
                          http://www.springframework.org/schema/rabbit http://www.springframework.org/schema/rabbit/spring-rabbit.xsd">
	<!--连接工厂-->
	<!--界面上连接的是15672,这是图形界面的窗口,这里5672是通信端口-->
	<rabbit:connection-factory id="connectionFactory" host="127.0.0.1" port="5672" username="guest" password="guest" publisher-confirms="true"/>
	<!--连接工厂,使用模板进行消息发送-->
	<rabbit:template id="rabbitTemplate" connection-factory="connectionFactory" />
</beans>
</beans>
 

(3)编写测试代码
获得模板进行消息发送

public class rabbitmqDemo {
    public static void main(String[] args) {
        /*加载配置文件*/
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext-rabbitmq-producer.xml");
        /*获得模板,实际使用中,直接注入就行*/
        RabbitTemplate rabbitTemplate = (RabbitTemplate) context.getBean("rabbitTemplate");
        rabbitTemplate.convertAndSend("", "queue.test", "直接模式");
        context.close();
    }
 
}
 
 

执行后观察控制台。

2.1.4 代码实现 - 消息消费者

(1)编写消息监听类

public class MessageConsumer implements MessageListener {
    public void onMessage(Message message) {
        /*message.getBody()返回的是字节数组*/
        System.out.println("收到消息:"+new String(message.getBody()));
    }
    public void containerAckMode(AcknowledgeMode mode) {
    }
}
 
 

(2)创建配置文件 applicationContext-rabbitmq-consumer.xm

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	   xmlns:rabbit="http://www.springframework.org/schema/rabbit"
	   xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
                          http://www.springframework.org/schema/rabbit http://www.springframework.org/schema/rabbit/spring-rabbit.xsd">
	<!--连接工厂-->
	<rabbit:connection-factory id="connectionFactory" host="127.0.0.1" port="5672" username="guest" password="guest" publisher-confirms="true"/>
	<!--队列-->
	<rabbit:queue name="queue.test" durable="true" exclusive="false" auto-delete="false" />
	<!--消费者监听类-->
	<bean id="messageConsumer" class="com.tech.verify.MessageConsumer"></bean>
	<!--设置监听容器,配置监听器监听哪个队列-,连接哪个工厂,哪个监听类,哪个队列-->
	<rabbit:listener-container connection-factory="connectionFactory" acknowledge="auto" >
		<rabbit:listener queue-names="queue.test" ref="messageConsumer"/>
	</rabbit:listener-container>
</beans>
 
 

(3)编写测试代码 Test2

public class StartListen {
    public static void main(String[] args) {
        /*当启动时自动是监听状态,自动寻找监听类,接收消息并执行*/
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext-rabbitmq-consumer.xml");
    }
}
 
收到消息:直接模式

消息一直处于监听状态, 此时查看消息队列已经无消息, 已经被取出
那么有多个消费者呢?
消息只有一个, 被某个消费者取走, 其他消费者就没有消息, 就像抢红包, 抢了就没了
绑定的路由 key 不存在, 则消息直接丢弃, 就像没有收件人地址的信

2.2 分列模式(Fanout)

群发模式

2.2.1 什么是分列(Fanout)模式

当我们需要将消息一次发给多个队列时,需要使用这种模式。如下图:

任何发送到 Fanout Exchange 的消息都会被转发到与该 Exchange 绑定 (Binding) 的所有 Queue 上。

  1. 这种模式需要提前将 Exchange 与 Queue 进行绑定,一个 Exchange 可以绑定多个 Queue,一个 Queue 可以同多个 Exchange 进行绑定。
  2. 这种模式不需要 RouteKey
  3. 如果接受到消息的 Exchange 没有与任何 Queue 绑定,则消息会被抛弃。

实际场景:
项目中数据库中的数据为了提升访问速度, 做了商品展示页静态化,
搜索的数据在数据库和 es 中存了两份,
商品修改交换器, 建立多个队列绑定, 商品导入索引库的队列, 静态化队列, 奖励两个工程, 那么当商品变更时发给交换器, 交换器发给队列, 队列执行相应的功能

2.2.2 交换器绑定队列

创建队列 queue.test1 和 queue.test2

创建交换器 exchange.fanout_test,并绑定 queue.test1 和 queue.test2

前面不是非要写点, 只是为了介绍身份
绑定

交换器也能发给交换器

2.2.3 代码实现 - 消息生产者

指定交换器而无路由 key
编写代码

public class rabbitmqDemo111 {
    public static void main(String[] args) {
        /*加载配置文件*/
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext-rabbitmq-producer.xml");
        /*获得模板,实际使用中,直接注入就行*/
        RabbitTemplate rabbitTemplate = (RabbitTemplate) context.getBean("rabbitTemplate");
        rabbitTemplate.convertAndSend("exchange.fanout_test", "", "分列模式");
        context.close();
    }
}
 

测试代码,发现队列 queue.test1 和 queue.test2 都收到消息

2.2.4 代码实现 - 消息消费者

(1)将 MessageConsumer 复制为 MessageConsumer1 和 MessageConsumer2。
(2)修改 applicationContext-rabbitmq-consumer.xml,增加配置

	<!--连接工厂-->
	<rabbit:connection-factory id="connectionFactory" host="127.0.0.1" port="5672" username="guest" password="guest" publisher-confirms="true"/>
	<!--队列-->
	<rabbit:queue name="queue.test" durable="true" exclusive="false" auto-delete="false" />
	<rabbit:queue name="queue.test1" durable="true" exclusive="false" auto-delete="false" />
	<rabbit:queue name="queue.test2" durable="true" exclusive="false" auto-delete="false" />
	<!--消费者监听类-->
	<bean id="messageConsumer" class="com.tech.verify.MessageConsumer"></bean>
	<bean id="messageConsumer1" class="com.tech.verify.MessageConsumer1"></bean>
	<bean id="messageConsumer2" class="com.tech.verify.MessageConsumer2"></bean>
	<!--设置监听容器,配置监听器监听哪个队列-,连接哪个工厂,哪个监听类,哪个队列-->
	<rabbit:listener-container connection-factory="connectionFactory" acknowledge="auto" >
		<rabbit:listener queue-names="queue.test" ref="messageConsumer"/>
		<rabbit:listener queue-names="queue.test1" ref="messageConsumer1"/>
		<rabbit:listener queue-names="queue.test2" ref="messageConsumer2"/>
	</rabbit:listener-container>
</beans>
 

测试:启动 Test2 测试, 不需要修改, 因为仅仅只是加载配置文件

222收到消息:分列模式
111收到消息:分列模式

2.1.5 创建队列与交换器(配置方式)

可以不必依赖界面,而是自动完场创建
发消息之前自动创建绑定
修改配置文件 applicationContext-rabbitmq-producer.xml

<!--rabbitAdmin 封装管理操作-->
	<!--在配置文件中涉及到rabbitmq的管理操作,据此获得权限,里面封装了许多操作-->
	<!--也可以指定id,这样就可以通过代码方式管理,此处未指定id-->
	<rabbit:admin connection-factory="connectionFactory"></rabbit:admin>
		<!--创建队列-->
		<rabbit:queue name="queue.test1" />
		<!--创建队列-->
		<rabbit:queue name="queue.test2" />
		<!--创建分发交换器,不同的交换器有不同的标签 -->
		<rabbit:fanout-exchange name="exchange.fanout_test" >
			<rabbit:bindings>
				<rabbit:binding queue="queue.test1"></rabbit:binding>
				<rabbit:binding queue="queue.test2"></rabbit:binding>
			</rabbit:bindings>
		</rabbit:fanout-exchange>
 

3. 用户注册

3.1 需求分析

注册账号,用手机号注册,填写后发送短信验证码,填写短信验证码正确方可注册成
功。

3.2 实现思路

(1)发送短信验证码逻辑:
用户服务模块将要发送的短信验证码发送给 rabbitmq 和 redis ,
短信服务模块将消息从 rabbitmq 中取出并调用阿里云通信发送短信。阿里云通信整合了三大运营商的短信网关,最终把验证码发送到用户的手机上。
rabbitmq 采用直接模式,用户服务为消息生产者,短信服务为消息消费者。

使用RabbitMQ作为消息中间件来调用阿里云平台发送短信验证码可以带来以下好处:

  1. 异步处理:通过将消息发送到RabbitMQ队列中,可以实现异步处理。用户服务模块将消息发送到RabbitMQ后,可以立即继续处理其他任务,而不必等待短信发送的完成。这可以提高系统的响应速度和吞吐量。

  2. 解耦和灵活性:将用户服务模块与短信服务模块解耦,它们可以独立进行扩展和部署。用户服务模块只需将消息发送到RabbitMQ,而不需要关心短信发送的具体实现细节。短信服务模块从RabbitMQ中获取消息后,可以根据需要选择适合的方式调用阿里云通信发送短信,而不会影响用户服务模块的运行。

  3. 提高可靠性:RabbitMQ具有消息持久化的能力,可以确保即使在发生故障或重启后,消息仍然可靠地传递给短信服务模块。这样可以避免消息的丢失或重复发送。

  4. 消息队列的特性:RabbitMQ提供了一些有用的特性,例如消息的确认机制、消息的路由和筛选、消息的优先级等。这些特性可以帮助实现更可靠和灵活的消息传递。

综上所述,使用RabbitMQ作为调用阿里云平台发送短信的中间件,可以提高系统的响应速度、可扩展性和可靠性,同时也能够降低模块之间的耦合度,使系统更加灵活和可维护。

(2)注册逻辑:注册时从 redis 中提取短信验证码与用户填写的验证码进行比对,如果一致则可以注册,否则拦截请求。

3.3 后端代码

首先生成用户服务,并添加到当前工程。

3.3.1 发送短信验证码到 MQ

实现思路: 在用户服务编写 API , 生成手机验证码,存入 Redis 并发送到 RabbitMQ
(1)因为要用到消息队列,所以在用户服务引入 rabbit 与 spring 的整合依赖。

<dependency>
<groupId>org.springframework.amqp</groupId>
<artifactId>spring‐rabbit</artifactId>
<version>2.1.4.RELEASE</version>
</dependency>
 

(2)添加配置文件 applicationContext-rabbitmq-producer.xml

<?xml version="1.0" encoding="UTF‐8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema‐instance"
        xmlns:rabbit="http://www.springframework.org/schema/rabbit"
        xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring‐beans.xsd
        http://www.springframework.org/schema/rabbit
        http://www.springframework.org/schema/rabbit/spring‐rabbit.xsd">
<!‐‐连接工厂‐‐>
<rabbit:connection‐factory id="connectionFactory" host="127.0.0.1"  port="5672" username="guest" password="guest" />
<rabbit:admin connection‐factory="connectionFactory"></rabbit:admin>
<!‐‐创建队列‐‐>
<rabbit:queue name="queue.sms" />
<rabbit:template id="rabbitTemplate" connection‐factory="connectionFactory" />
</beans>
 
 

(3)在 UserService 中新增方法定义

/**
 * 发送短信验证码
 * @param mobile
 */
public void sendSms(String phone);
 

(4)UserServiceImpl 方法实现

@Autowired
private RedisTemplate redisTemplate;
@Autowired
private RabbitTemplate rabbitTemplate;
/**
 * 发送短信验证码
 * @param phone
 */
public void sendSms(String phone){
//1.得到六位短信验证码
        int max=999999;
        int min=100000;
        Random random = new Random();
        int code = random.nextInt(max);
        //保证验证码为6位
        if(code<min){
        code=code+min;
        }
        System.out.println("短信验证码:"+code);
//2.保存到redis里
        redisTemplate.boundValueOps("code_"+phone).set(code+"");
        redisTemplate.boundValueOps("code_"+phone).expire(5,TimeUnit.MINUTES);//5分钟失效
//3.发送给RabbitMQ
        Map<String,String> map=new HashMap();
        map.put("phone", phone);
        map.put("code", code+"");
        rabbitTemplate.convertAndSend("","queue.sms",
        JSON.toJSONString(map));
        }
 

(5)qingcheng_web_portal 新增 UserController

@RestController
@RequestMapping("/user")
public class UserController {
    @Reference
    private UserService userService;
    /**
     * 发送短信验证码
     * @param phone
     */
    @GetMapping(value="/sendSms")
    public Result sendSms(String phone){
        userService.sendSms(phone);
        return new Result();
    }
}
 

3.3.2 短信服务接收消息

短信发送是由单独的短信服务提供的功能,所有的短信都是先发送到消息队列,短
信服务从消息队列中提取手机号和验证码,调用短信发送接口进行发送短信

我们这个环节实现的是将手机号和验证码从消息队列中提取出来,打印到控制台上。
(1)创建 qingcheng_service_sms 工程,pom 文件引入依赖

<dependency>
<groupId>org.springframework.amqp</groupId>
<artifactId>spring‐rabbit</artifactId>
<version>2.1.4.RELEASE</version>
</dependency>
<dependency>
<groupId>com.qingcheng</groupId>
<artifactId>qingcheng_common</artifactId>
<version>1.0‐SNAPSHOT</version>
</dependency>
 

(2)添加 web.xml,参见其它服务工程
(3)添加监听实现类 SmsMessageConsumer

public class SmsMessageConsumer implements MessageListener {
    public void onMessage(Message message) {
        String jsonString = new String(message.getBody());
        Map<String,String> map = JSON.parseObject(jsonString, Map.class);
        String phone = map.get("phone");
        String code=map.get("code");
        System.out.println("手机号:"+phone+"验证码:"+code);
    }
}
 

(4)添加配置文件 applicationContext-rabbitmq-consumer.xml

<!‐‐连接工厂‐‐>
<rabbit:connection‐factory id="connectionFactory" host="127.0.0.1"
        port="5672" username="guest" password="guest" />
<!‐‐创建队列‐‐>
<rabbit:queue name="queue.sms" />
<!‐‐消费者监听类‐‐>
<bean id="messageConsumer"
class="com.qingcheng.consumer.SmsMessageConsumer"></bean>
<!‐‐设置监听容器‐‐>
<rabbit:listener‐container connection‐factory="connectionFactory" >
<rabbit:listener queue‐names="queue.sms" ref="messageConsumer"/>
</rabbit:listener‐container>
 
 

3.3.3 用户注册

(1)UserService 增加方法定义

/**
 * 增加
 * @param user
 * @param smsCode
 */
public void add(User user,String smsCode);
 

(2)UserServiceImpl 实现方法

/**
 * 增加
 * @param user
 * @param smsCode
 */
public void add(User user,String smsCode) {
//比较短信验证码
//获取系统短信验证码
        String sysCode= (String)
        redisTemplate.boundValueOps("code_"+user.getPhone()).get();
        if(sysCode==null){
        throw new RuntimeException("验证码未发送或已过期");
        }
        if(!smsCode.equals(sysCode)){
        throw new RuntimeException("验证码不正确");
        }
        if(user.getUsername()==null){
        user.setUsername(user.getPhone());
        }
        User searchUser=new User();
        searchUser.setUsername(user.getUsername());
        if(userMapper.selectCount(searchUser)>0) { //查询是否存在相同记录
        throw new RuntimeException("该手机号已注册");
        }
        user.setCreated(new Date());
        user.setUpdated(new Date());
        user.setPoints(0);//积分初始值为0
        user.setStatus("1");//状态1
        user.setIsEmailCheck("0");//邮箱认证
        user.setIsMobileCheck("1");//手机认证
        userMapper.insert(user);
        }
 

(3)UserController 增加方法

@PostMapping("/save")
public Result save(@RequestBody User user , String smsCode ){
//密码加密
        BCryptPasswordEncoder encoder=new BCryptPasswordEncoder();
        String newpassword = encoder.encode(user.getPassword());
        user.setPassword(newpassword);
        userService.add(user,smsCode);
        return new Result();
        }
 
 

3.4 前端代码

(1)写 register.html 到 qingcheng_web_portal 的 webapp 下,在 body 中放置 div

(2)表单部分代码绑定

<div class="control‐group">
<label class="control‐label">手机号:</label>
<div class="controls">
<input type="text" placeholder="请输入你的手机号" v‐
        model="pojo.phone" class="input‐xfat input‐xlarge">
</div>
</div>
<div class="control‐group">
<label class="control‐label">验证码:</label>
<div class="controls">
<input type="text" placeholder="验证码" v‐model="smsCode"
class="input‐xfat input‐xlarge">
<a @click="sendSms()" >发送短信验证码</a>
</div>
</div>
<div class="control‐group">
<label class="control‐label">登录密码:</label>
<div class="controls">
<input type="password" placeholder="设置登录密码" v‐
        model="pojo.password" class="input‐xfat input‐xlarge">
</div>
</div>
<div class="control‐group">
<label class="control‐label">确认密码:</label>
<div class="controls">
<input type="password" placeholder="再次确认密码" v‐
        model="password" class="input‐xfat input‐xlarge">
</div>
</div>
<div class="control‐group">
<label class="control‐label">      
</label>
<div class="controls">
<input name="m1" type="checkbox" value="2" checked=""><span>同意协议并注册《用户协议》</span>
</div>
</div>
<div class="control‐group">
<label class="control‐label"></label>
<div class="controls btn‐reg">
<a class="sui‐btn btn‐block btn‐xlarge btn‐danger"
@click="save()" >完成注册</a>
</div>
</div>
 

(3)编写 js 部分代码

<script src="/js/vue.js"></script>
<script src="/js/axios.js"></script>
<script>
var vue=new Vue({
        el: '#app',
        data(){
        return {
        pojo: {},
        smsCode:"",
        password:""
        }
        },
        methods:{
        sendSms(){
        axios.get(`/user/sendSms.do?
        phone=${this.pojo.phone}`).then(response => {
        alert(response.data.message);
        });
        },
        save (){
        if(this.password!=this.pojo.password){
        alert("两次输入密码不一致");
        return ;
        }
        axios.post(`/user/save.do?
        smsCode=${this.smsCode}`,this.pojo).then(response => {
        alert(response.data.message);
        this.pojo={};
        this.smsCode="";
        this.password="";
        });
        }
        }
        })
</script>
 

4. 阿里云通信

4.1 阿里云通信简介

阿里云通信(原名阿里大于),是阿里云旗下产品,融合了三大运营商的通信能力,提供包括短信、语音、流量直充、私密专线、店铺手机号等个性化服务。

4.2 发短信前你要准备什么

(1)在阿里云官网 https://www.aliyun.com/ 注册账号
(2)完成实名认证
(3)登陆阿里云,产品中选择” 短信服务 “,直接搜索即可
(4)申请签名(选择验证码类型),每个短信上都会标注来源,国内消息
(5)申请模板
(6)创建 accessKey (注意保密!),点击个人头像
(7)充值
查看 api demo

4.3 快速入门

(1)创建工程引入依赖

<dependency>
<groupId>com.aliyun</groupId>
<artifactId>aliyun‐java‐sdk‐core</artifactId>
<version>4.0.3</version>
</dependency>
 

(2)创建测试类,以下代码从官网获取

import com.aliyuncs.CommonRequest;
import com.aliyuncs.CommonResponse;
import com.aliyuncs.DefaultAcsClient;
import com.aliyuncs.IAcsClient;
import com.aliyuncs.exceptions.ClientException;
import com.aliyuncs.exceptions.ServerException;
import com.aliyuncs.http.MethodType;
import com.aliyuncs.profile.DefaultProfile;
/*
pom.xml
<dependency>
<groupId>com.aliyun</groupId>
<artifactId>aliyun‐java‐sdk‐core</artifactId>
<version>4.0.3</version>
</dependency>
*/
public class CommonRpc {
    public static void main(String[] args) {
    //ssk 账户id,秘钥
        DefaultProfile profile = DefaultProfile.getProfile("cn‐hangzhou","*******", "********");
        IAcsClient client = new DefaultAcsClient(profile);
        CommonRequest request = new CommonRequest();
//request.setProtocol(ProtocolType.HTTPS);
        request.setMethod(MethodType.POST);
        request.setDomain("dysmsapi.aliyuncs.com");
        request.setVersion("2020‐05‐25");
        request.setAction("SendSms");
        request.putQueryParameter("RegionId", "cn‐hangzhou");
        request.putQueryParameter("PhoneNumbers", "175********");
        request.putQueryParameter("SignName", "青橙");
        request.putQueryParameter("TemplateCode", "SMS_165116876");
        request.putQueryParameter("TemplateParam", "
        {\"code\":\"123123\"}");
            try {
                CommonResponse response = client.getCommonResponse(request);
                System.out.println(response.getData());
            } catch (ServerException e) {
                e.printStackTrace();
            } catch (ClientException e) {
                e.printStackTrace();
                }
}
}
 

4.4 短信服务发送短信

(1)短信服务 pom.xml 引入阿里云 sdk 依赖

<dependency>
<groupId>com.aliyun</groupId>
<artifactId>aliyun‐java‐sdk‐core</artifactId>
<version>4.0.3</version>
</dependency>
 
 

(2)添加配置文件 sms.properties

accessKeyId=******
        accessKeySecret=*******
        smsCode=SMS_165116876
        param={"code":"[value]"}
 
 

(3)添加配置文件 applicationContext-sms.xml

<context:component‐scan base‐package="com.qingcheng"></context:component‐scan>
 

(4)创建短信工具类 SmsUtil

@Component
public class SmsUtil {
    @Value("${accessKeyId}")
    private String accessKeyId;
    @Value("${accessKeySecret}")
    private String accessKeySecret;
    public CommonResponse sendSms(String phone,String smsCode,String
            param){
        DefaultProfile profile = DefaultProfile.getProfile("cn‐hangzhou", accessKeyId, accessKeySecret);
        IAcsClient client = new DefaultAcsClient(profile);
        CommonRequest request = new CommonRequest();
//request.setProtocol(ProtocolType.HTTPS);
        request.setMethod(MethodType.POST);
        request.setDomain("dysmsapi.aliyuncs.com");
        request.setVersion("2020‐05‐25");
        request.setAction("SendSms");
        request.putQueryParameter("RegionId", "cn‐hangzhou");
        request.putQueryParameter("PhoneNumbers", phone);
        request.putQueryParameter("SignName", "青橙");
        //模板编号,参数
        request.putQueryParameter("TemplateCode", smsCode);
        request.putQueryParameter("TemplateParam", param);
        try {
            CommonResponse response = client.getCommonResponse(request);
            System.out.println(response.getData());
            return response;
        } catch (ServerException e) {
            e.printStackTrace();
            return null;
        } catch (ClientException e) {
            e.printStackTrace();
            return null;
        }
    }
}
 

(5)修改消息监听类,完成短信发送

public class SmsMessageConsumer implements MessageListener {
    @Autowired
    private SmsUtil smsUtil;
    @Value("${smsCode}")
    private String smsCode;//短信模板编号
    @Value("${param}")
    private String param;//短信参数
    public void onMessage(Message message) {
        String jsonString = new String(message.getBody());
        Map<String,String> map = JSON.parseObject(jsonString, Map.class);
        String phone = map.get("phone");
        String code=map.get("code");
        System.out.println("手机号:"+phone+"验证码:"+code);
        String param= templateParam_smscode.replace("[value]",code);
        try {
            SendSmsResponse smsResponse = smsUtil.sendSms(phone, smsCode,
                    param);
        } catch (ClientException e) {
            e.printStackTrace();
        }
    }
}
 

5. 商品上下架消息处理

5.1 需求分析

在商品上架后,生成商品详细页和新增 elasticsearch 索引数据。
在商品下架后,删除商品详细页和删除 elasticsearch 数据.
以上操作采用消息中间件 rabbitmq 解耦调用

5.2 实现思路

因为有多种业务逻辑需要处理,所以我们需要使用分列模式来处理请求。
(1)创建两个服务工程
一是商品详细页生成服务(qingcheng_service_page),对上架的商品重新生成上新品详细页。
二是索引数据更新服务(qingcheng_service_index),对上架的商品重新导入索引库。
(2)通过配置文件实现:
rabbitmq 新增两个交换器 ,分别是商品上架交换器和商品下架交换器。
rabbitmq 新增商品详细页生成队列、商品详细页删除队列、elasticsearch 数据新增队列、elasticsearch 数据删除队列。
商品上架交换器绑定新增商品详细页生成队列和 elasticsearch 数据新增队列
商品下架交换器绑定商品详细页删除队列和 elasticsearch 数据删除队列
(3)在商品服务的上架发送消息到 rabbitmq 的商品上架交换器,在商品服务的下架发送消息到 rabbitmq 的商品下架交换器
(4)商品详细页生成服务从 新增商品详细页生成队列和商品详细页删除队列提取消息进行逻辑处理
(5)索引数据更新服务从 elasticsearch 数据新增队列和 elasticsearch 数据删除队列提取消息进行逻辑处理。