mqtt协议理解
2015-12-10 17:12
337 查看
mqtt协议包格式
控制包最多由3部分组成:固定头部、可变头部和payload。
1. 固定头部
固定头部只有两个字节,格式如下:
byte1的高4位就是包的类型,包的类型定义如下:
低4位为flag位,有的包需要,有的包不需要:
byte2为剩余数据的长度,数据是包含可变头部和payload的。这个长度如果是单个字节的话,最多只能是127,如果数值大于127怎么办,则用多个字节来表示,编码方案是,低7位用来表示数值,最高位是一个"continuation bit",表示后面是否还有长度字节,例如:如果是64,那么只需要一个字节就可以表示,十六进制为0x40。如果是321(65+2*128),需要用两个字节来表示,那么第一个字节为65+128=192(注1100 0001),第2个字节为2(注0000 0010),有点像进位。但是这个length不是任意的,最多4个字节,所以最多只能表示为0xFF、0xFF、0xFF、0x7F,表示长度最大只能为268,435,455。
假设给了一个非负整数X,那么这个可变长度的编码算法如下:
那么相应的解码算法如下:
2. 可变头部和payload
可变头部和payload相对于就比较简单了,只是部分类型的包才需要:
控制包最多由3部分组成:固定头部、可变头部和payload。
1. 固定头部
固定头部只有两个字节,格式如下:
byte1的高4位就是包的类型,包的类型定义如下:
低4位为flag位,有的包需要,有的包不需要:
byte2为剩余数据的长度,数据是包含可变头部和payload的。这个长度如果是单个字节的话,最多只能是127,如果数值大于127怎么办,则用多个字节来表示,编码方案是,低7位用来表示数值,最高位是一个"continuation bit",表示后面是否还有长度字节,例如:如果是64,那么只需要一个字节就可以表示,十六进制为0x40。如果是321(65+2*128),需要用两个字节来表示,那么第一个字节为65+128=192(注1100 0001),第2个字节为2(注0000 0010),有点像进位。但是这个length不是任意的,最多4个字节,所以最多只能表示为0xFF、0xFF、0xFF、0x7F,表示长度最大只能为268,435,455。
假设给了一个非负整数X,那么这个可变长度的编码算法如下:
do encodeByte = X MOD 128 X = X DIV 128 // if there are more data to encode, set the top bit of this byte if (X > 0) encodeByte = encodeByte OR 128 endif 'output' encodeByte while (X > 0)我写的c测试代码如下:
#include <stdio.h> #include <inttypes.h> int main(void) { uint8_t len_byte[4] = {0x00, 0x00, 0x00, 0x00}, encodeByte; uint32_t X = 321; int i = 0; do { encodeByte = X % 128; X = X / 128; if (X > 0) { encodeByte = encodeByte | 128; } len_byte[i++] = encodeByte; } while (X > 0); return 0; }
那么相应的解码算法如下:
multiplier = 1 value = 0 do encodeByte = 'next byte from stream' value += (encodeByte AND 127) * multiplier multiplier *= 128 if (multiplier > 128*128*128) throw Error(Malformed Remaining Length) while ((encodeByte AND 128) != 0)测试c代码如下:
#include <stdio.h> #include <inttypes.h> int main(void) { uint8_t len_byte[4] = {193, 2, 0, 0}, encodeByte; int multiplier, value, i; multiplier = 1; value = 0; i = 0; do { encodeByte = len_byte[i++]; value += (encodeByte & 127) * multiplier; multiplier *= 128; if (multiplier > 128*128*128) { printf("Malformed Remaining Length\n"); } } while ((encodeByte & 128) != 0); return 0; }
2. 可变头部和payload
可变头部和payload相对于就比较简单了,只是部分类型的包才需要:
相关文章推荐
- 自动化测试培训:qtp脚本获取获取汇率数据
- Qt浅谈之三十六仿360设置中心
- QT获取控件中心点,图片按比例缩放
- QT检查文件是否存在
- Qt交叉编译imx6
- Qt实现指定线程执行回调
- 老李分享:QTP的录制原理以及实现
- Qt安装和QML HelloWord程序
- QT Creator安装版跟进 QT源码笔记
- 可供VC调用的QT编写的界面DLL方法
- Qt入门-layout布局
- QT 系统托盘图标
- [Qt]自定义控件之开关控件
- Qt使用教程之添加编译器(二)
- QT打包发布
- pyqt右键菜单
- Qt信息隐藏(Q_D/Q_Q)介绍——从二进制兼容讲起
- Qt中如果通过QStyle自定义能够跨平台的界面控件
- QT中静态库的生成与使用——创建共享库代码,附例子
- 调色板类QPalette——包含了Qt窗口不见的颜色组(collor group),和Windows右键属性外观非常类似