【IM即时通讯】MQTT协议的详解

4 篇文章 0 订阅
4 篇文章 0 订阅

【IM即时通讯】MQTT协议的详解

本博客大致内容均来自:MQTT官方文档
在这里插入图片描述



一、MQTT 表示报文类型用了 4 位,16 种组合。

在MQTT协议中,一个MQTT数据包由:固定头(Fixed header)、可变头(Variable header)、消息体(payload)三部分构成。

报文类型含义
Fixed Header固定报文头
Variable Header可变报文头 (注意这个是非必须的)
PayLoad荷载(也就是数据)(注意这个是非必须的)

红色部分为固定头、绿色可变头、紫色为消息体
在这里插入图片描述

(1)固定头(Fixed header)。

  • 【Message Type】存在于所有MQTT数据包中,表示数据包类型及数据包的分组类标识。
NameValue(2进制示例)Direction of flowDescription
Reserved0禁止占位不使用
CONNECT0001 0000Client->Server客户端连接时候使用,例如PC\APP
CONNACK0010 0000Server->Client连接完毕后进行ACK确认
PUBLISH0011 0000<->发布消息(发送消息)示例中DUP、两个QoS、RETAIN全填0为例
PUBACK0010 0000<->QoS1消息发布确认
PUBREC0011 0000<->发布收到
PUBREL0101 0000<->发布释放
PUBCOMP0111 0000<->Qos2消息发布完成
SUBSCRIBE1000 0010Client->Server客户端订阅需求
SUBACK1001 0000Server->Client订阅请求报ACK
UNSUBSCRIBE1010 0010Client->Server客户端取消订阅请求
UNSUBACK1011 0000Server->Client取消订阅请求ACK
PINGREQ1100 0000Client->Server心跳请求
PINGRESP1110 0000Server->Client心跳ACK
DISCONNECT1111 0000Client->Server客户端连接时候使用,例如PC\APP
Reserved15禁止占位不使用
  • 【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禁止0000
CONNACK禁止0000
PUBLISHMQTT 3.1.1 中使用DUP1QoS2QoS2RETAIN3
PUBACK禁止0000
PUBREC禁止0000
PUBREL禁止0010
PUBCOMP禁止0000
SUBSCRIBE禁止0010
SUBACK禁止0000
UNSUBSCRIBE禁止0010
UNSUBACK禁止0000
PINGREQ禁止0000
PINGRESP禁止0000
DISCONNECT禁止0000
  • 【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 一次阐述不同类型下所对应的数据内容大致是怎么分布在各个字节的
  • 15
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值