1.MQTT协议简述

MQTT(消息队列遥测传输协议),是一种基于发布/订阅模式的轻量级通讯协议,构建于TCP/IP协议上。MQTT最大的优点在于,用极少的代码和有效的带宽,为连接远程设备提供实时可靠的消息服务

发布订阅模式(Publish-Subscribe Pattern)是一种消息传递模式,它将发送消息的客户端(发布者)与接收消息的客户端(订阅者)解耦。MQTT订阅/发布模型主要由一个称为代理(Broker)的中间角色负责所有消息的路由和分发工作,发布者将带有主题的消息发送给代理,订阅者则向代理订阅主题来接收感兴趣的消息。

在MQTT中,主题和订阅无法被提前注册或创建,所以代理也无法预知某一个主题之后是否会有订阅者,以及会有多少订阅者,所以只能将消息转发给当前的订阅者,如果当前不存在任何订阅,那么消息将被直接丢弃。

MQTT有4个主要组成部分:发布者,订阅者,代理和主题

发布者(Publisher)

负责将消息发布到主题上,发布者一次只能向一个主题发送数据,发布者发布消息时无需关注订阅者是否在线。

订阅者(Subscriber)

订阅者通过订阅主题接收消息,且可一次订阅多个主题。MQTT还支持通过共享订阅的方式在多个订阅者之间实现订阅的负载均衡

代理(Broker)

负责接收发布者的消息,并将消息转发至符合条件的订阅者。另外,代理也需要负责处理客户端发起的连接、断开连接、订阅、取消订阅等请求

主题(Topic)

主题是MQTT进行消息路由的基础,它类似URL路径,使用斜杠/进行分层,比如sensor/1/temperature。一个主题可以有多个订阅者,代理会将主题下的消息转发给所有的订阅者;一个主题也可以有多个发布者,代理将按照消息到达的顺序转发。

MQTT发布/订阅中的消息路由

在MQTT中,一个客户端既可以是发布者,也可以是订阅者,也可以同时具备这两个身份。当客户端发布一条消息时,它会被发送到代理,然后代理将消息路由到该主题的所有订阅者。当客户端订阅一个主题时,它会受到代理转发到该主题的所有消息。

大多数发布/订阅系统主要通过以下两种方式过滤并路由消息:

  • 根据主题

订阅者向代理订阅自己感兴趣的主题,发布者发布所有的消息中都会包含自己的主题,代理根据消息的主题判断需要将消息转发给哪些订阅者

  • 根据消息内容

订阅者定义其感兴趣的消息的条件,只有当消息的属性或内容满足订阅者定义的条件时,协议才会被投递到该订阅者。

MQTT协议是基于主题进行消息路由的。

MQTT与HTTP请求响应

HTTP协议相比MQTT需要占用更多的网络开销,且HTTP是一种无状态协议,服务器在处理请求时不会记录客户端的状态,也无法实现从连接异常断开中恢复;请求响应模式需要通过轮询才能获取数据更新,而MQTT通过订阅即可获取实时数据更新。

发布订阅模式的松耦合特性,也有一些副作用。由于发布者不知晓订阅者的状态,因此发布者无法得知订阅者是否收到了消息,或者是否正确处理了消息,为此,MQTT5.0增加了请求响应特性,以实现订阅者收到消息后向某个主题发送应答,发布者收到应答后再进行后续操作。

MQTT与消息队列

MQTT与消息队列的很多行为和特性非常接近,都采用发布/订阅模式,但是使用场景明显不同。消息队列主要用于服务端应用之间的消息存储与转发,这类场景数据量大但客户端数量少。MQTT是一种消息传输协议,主要用于物联网设备之间的消息传递,这类场景的特点是海量的设备接入,管理与消息传输。

在实际的应用场景中,MQTT与消息队列会结合使用,以使MQTT服务器能专注于处理设备的连接与设备间的消息路由。比如先由MQTT服务器接收物联网设备上报的数据,然后再通过消息队列将这些数据转发到不同业务系统进行处理。

MQTT主题不需要提前创建,MQTT客户端在订阅或发布时即自动创建了主题,开发者无需再关心主题的创建,也不需要手动删除主题。

MQTT协议的特点

MQTT是一个基于客户端-服务器的消息发布/订阅传输协议。

MQTT与HTTP一样,MQTT运行在传输控制协议/互联网协议(TCP/IP)之上。

java mqtt取消订阅 mqtt订阅消息_消息队列

发布订阅

MQTT使用发布/订阅消息模式,它提供了一对多的消息分发机制,从而实现与应用程序的解耦。

这是一种消息传递模式,消息不是直接从发送器发送到接收器(点对点),而是由MQTT Server(MQTT Broker)分发的。

java mqtt取消订阅 mqtt订阅消息_消息队列_02

QoS

服务质量是MQTT的一个重要特性,当我们使用TCP/IP时,连接已经在一定程度上受到保护。再试在无线网络中,中断和干扰很频繁,MQTT可以帮助避免信息丢失及其服务质量水平。

QoS 0

这一级别会发生消息丢失或重复,消息发布依赖于底层TCP/IP网络。即:<=1

java mqtt取消订阅 mqtt订阅消息_客户端_03

QoS 1

QoS 1承诺消息将至少传送一次给订阅者

QoS 2

使用QoS2,保证消息仅传送到目的地一次。为此带有唯一消息ID的消息会存储两次,首先来自于发送者,然后是接收者。QoS级别2在网络中具有最高的开销,因为在发送方和接收方之间需要两个流。

java mqtt取消订阅 mqtt订阅消息_服务器_04

2.MQTT数据包结构

  • 固定头(Fixed header),存在于所有MQTT数据包中,表示数据包类型及数据包的分组类标识
  • 可变头(Variable header),存在于部分MQTT数据包中,数据包类型决定了可变头是否存在及其具体内容
  • 消息体(Payload),存在于部分MQTT数据包中,表示客户端收到的具体内容

整体MQTT消息格式:

java mqtt取消订阅 mqtt订阅消息_消息队列_05

2.1 MQTT固定头

固定头存在于所有的MQTT数据包中,结构:

java mqtt取消订阅 mqtt订阅消息_服务器_06

MQTT消息类型/Message type:

4位无符号值,类型如下:

标识符/DUP:

在不使用标识位的消息类型中,标识位被作为保留位。如果收到无效的标志时,接收端必须关闭网络连接:

  • DUP:发布消息的副本。用来在保证消息的可靠传输,如果设置为 1,则在下面的变长中增加MessageId,并且需要回复确认,以保证消息传输完成,但不能用于检测消息重复发送。
  • QoS发布消息的服务质量,即:保证消息传递的次数
  • 00:至多一次,即:<=1
  • 01:至少一次,即:>=1
  • 10:一次,即:=1
  • 11: 预留
  • RETAIN:发布保留标识,表示服务器要保留这次推送的信息,如果有新的订阅者出现,就把这消息推送给它,如果设有那么推送至当前订阅者后释放。

剩余长度:

固定头的第二字节用来保存变长头部和消息体的总大小的,但不是直接保存的。这一字节是可以扩展,其保存机制,前7位用于保存长度,后一部用做标识。当最后一位为 1时,表示长度不足,需要使用二个字节继续保存。例如:计算出后面的大小为0

2.2 MQTT可变头/Variable header

MQTT数据包中包含一个可变头,它驻位于固定的头和负载之间。可变头的内容因数据包类型而不同,较常的应用是做为包的标识:

很多类型数据包中都包括一个2字节的数据包标识字段,这些类型的包有:
PUBLISH (QoS > 0)、PUBACK、PUBREC、PUBREL、PUBCOMP、
SUBSCRIBE、SUBACK、UNSUBSCRIBE、UNSUBACK

2.3 PayLoad消息体

Payload消息体是MQTT数据包的第三部分,CONNECT、SUBSCRIBE、SUBACK、UNSUBSCRIBE四种类型的消息 有消息体:

  • CONNECT : 消息体内容主要是:客户端的ClientID、订阅的Topic、Message以及用户名和密码
  • SUBSCRIBE : 消息体内容是一系列的要订阅的主题以及QoS。
  • SUBACK : 消息体内容是服务器对于SUBSCRIBE所申请的主题及QoS进行确认和回复。
  • UNSUNSCRIBE : 消息体内容是要订阅的主题。










参考文章:

https://www.nxp.com.cn/products/processors-and-microcontrollers/s32-automotive-platform/s32g-vehicle-network-processors/s32g3-processors-for-vehicle-networking:S32G3

https://baijiahao.baidu.com/s?id=1757970960542623565&wfr=spider&for=pc