【IM即时通讯】MQTT协议的详解
本博客大致内容均来自:MQTT官方文档
一、MQTT 表示报文类型用了 4 位,16 种组合。
在MQTT协议中,一个MQTT数据包由:固定头(Fixed header)、可变头(Variable header)、消息体(payload)三部分构成。
报文类型 | 含义 |
---|---|
Fixed Header | 固定报文头 |
Variable Header | 可变报文头 (注意这个是非必须的) |
PayLoad | 荷载(也就是数据)(注意这个是非必须的) |
红色部分为固定头、绿色可变头、紫色为消息体
(1)固定头(Fixed header)。
- 【Message Type】存在于所有MQTT数据包中,表示数据包类型及数据包的分组类标识。
Name | Value(2进制示例) | Direction of flow | Description |
---|---|---|---|
Reserved | 0 | 禁止 | 占位不使用 |
CONNECT | 0001 0000 | Client->Server | 客户端连接时候使用,例如PC\APP |
CONNACK | 0010 0000 | Server->Client | 连接完毕后进行ACK确认 |
PUBLISH | 0011 0000 | <-> | 发布消息(发送消息)示例中DUP、两个QoS、RETAIN全填0为例 |
PUBACK | 0010 0000 | <-> | QoS1消息发布确认 |
PUBREC | 0011 0000 | <-> | 发布收到 |
PUBREL | 0101 0000 | <-> | 发布释放 |
PUBCOMP | 0111 0000 | <-> | Qos2消息发布完成 |
SUBSCRIBE | 1000 0010 | Client->Server | 客户端订阅需求 |
SUBACK | 1001 0000 | Server->Client | 订阅请求报ACK |
UNSUBSCRIBE | 1010 0010 | Client->Server | 客户端取消订阅请求 |
UNSUBACK | 1011 0000 | Server->Client | 取消订阅请求ACK |
PINGREQ | 1100 0000 | Client->Server | 心跳请求 |
PINGRESP | 1110 0000 | Server->Client | 心跳ACK |
DISCONNECT | 1111 0000 | Client->Server | 客户端连接时候使用,例如PC\APP |
Reserved | 15 | 禁止 | 占位不使用 |
- 【DUP、QOs Lvel、 RET】
**位置:**byte 1, bits 3-0。- DUP:发布消息的副本。用来在保证消息的可靠传输,如果设置为 1,则在下面的变长中增加MessageId,并且需要回复确认,以保证消息传输完成,但不能用于检测消息重复发送。
- QoS发布消息的服务质量(前面已经做过介绍),即:保证消息传递的次数
00:最多一次,即:<=1
01:至少一次,即:>=1
10:一次,即:=1
11:预留 - RETAIN:发布保留标识,表示服务器要保留这次推送的信息,如果有新的订阅者出现,就把这消息推送给它,如果设有那么推送至当前订阅者后释放。
标记为所对应的类型:表示
message Type (消息类型) | Fixed header flags(固定头标记位) | 第3位字节 | 第2位字节 | 第1位字节 | 第0位字节 |
---|---|---|---|---|---|
CONNECT | 禁止 | 0 | 0 | 0 | 0 |
CONNACK | 禁止 | 0 | 0 | 0 | 0 |
PUBLISH | MQTT 3.1.1 中使用 | DUP1 | QoS2 | QoS2 | RETAIN3 |
PUBACK | 禁止 | 0 | 0 | 0 | 0 |
PUBREC | 禁止 | 0 | 0 | 0 | 0 |
PUBREL | 禁止 | 0 | 0 | 1 | 0 |
PUBCOMP | 禁止 | 0 | 0 | 0 | 0 |
SUBSCRIBE | 禁止 | 0 | 0 | 1 | 0 |
SUBACK | 禁止 | 0 | 0 | 0 | 0 |
UNSUBSCRIBE | 禁止 | 0 | 0 | 1 | 0 |
UNSUBACK | 禁止 | 0 | 0 | 0 | 0 |
PINGREQ | 禁止 | 0 | 0 | 0 | 0 |
PINGRESP | 禁止 | 0 | 0 | 0 | 0 |
DISCONNECT | 禁止 | 0 | 0 | 0 | 0 |
- 【Remaining Length】
细心的同学还发现了一个 Byte 也在固定头中,这个为了 保存变长头部和消息体的总大小的,但不是直接保存的。这一字节是可以扩展,其保存机制,前7位用于保存长度,后一部用做标识。当最后一位为 1时,表示长度不足,需要使用二个字节继续保存。例如:计算出后面的大小为0
【敲黑板】
:
表示的是本数据包剩余部分的字节数,即可变头部和载荷的字节数之和。为了节省传输时的字节数,Remaining Length 采用的是一种变长编码方式。这就是说 Remaining Length 字段的字节数不是固定的,它可能使用1~4个字节。既然 Remaining Length 的字节数是可变的,那么问题来了,我们在解码包数据的时候,怎么知道 Remaining Length 究竟是使用几个字节编码的呢?解决这个问题的办法是,将每个字节的最高位(MSB)作为标志位。若该位的值是1,则意味着下一个字节属于参与 Remaining Length 编码的字节;若该位的值是0,则意味着本字节已经是最后一个参与 Remaining Length 编码的字节了。
在不使用标识位的消息类型中,标识位被作为保留位。如果收到无效的标志时,接收端必须关闭网络连接。
PS:
若当前读到的字节是: 0x50(0101 0000),则说明 Remaining Length 字段只用1个字节编码;
若连续读到的字节是:0x80(1000 0000), 0x80(1000 0000), 0x01(0000 0001)则说明 Remaining Length 字段占3个字节。
(2)可变头(Variable header)。存在于部分MQTT数据包中,数据包类型决定了可变头是否存在及其具体内容。
- 可变头部正如它的名字一样,是不定的,不同的包类型具有不同的可变头部。但许多包都有包 ID(Packet Identifier)字段。下表是包 ID 字段在各类型包中的存在情况
需要说明的是:Package ID 的 大小是 2字节
message Type (消息类型) | 是否要求存在包ID |
---|---|
CONNECT | 禁止 |
CONNACK | 禁止 |
PUBLISH | 如果说Qos等级大于0 需要 |
PUBACK | 需要 |
PUBREC | 需要 |
PUBREL | 需要 |
SUBSCRIBE | 需要 |
SUBACK | 需要 |
UNSUBSCRIBE | 需要 |
UNSUBACK | 需要 |
PINGREQ | 禁止 |
PINGRESP | 禁止 |
DISCONNECT | 禁止 |
(3)消息体(Payload)。存在于部分MQTT数据包中,表示客户端收到的真正内容。与可变头一样,在有些协议类型中有消息内容,有些协议类型中没有消息内容。
message Type (消息类型) | 是否必须要求存在载荷 |
---|---|
CONNECT | 需要 |
CONNACK | 不需要 |
PUBLISH | 可以为空 |
PUBACK | 不需要 |
PUBREC | 不需要 |
PUBREL | 不需要 |
SUBSCRIBE | 需要 |
SUBACK | 需要 |
UNSUBSCRIBE | 需要 |
UNSUBACK | 不需要 |
PINGREQ | 不需要 |
PINGRESP | 不需要 |
DISCONNECT | 不需要 |
后续
- 后续回根据不同的message Type 一次阐述不同类型下所对应的数据内容大致是怎么分布在各个字节的