当前位置: 首页 > news >正文

浅谈通信协议设计

目录

1.设计原则

2.注意事项

2.1.大小端编码问题

2.2.字节对齐

2.3.显式指定整型字段的长度

2.4.涉及到浮点数要考虑精度问题,建议放大成整数或者使用字符串去传输

2.5.协议与自动升级功能


1.设计原则

设计通信协议时,需要考虑以下几个原则:

1) 简单性:协议设计应尽可能简单,简单的协议更易于实现和维护,减少出错的可能性。复杂的协议往往会导致理解和实现的困难,增加维护成本。

2) 可扩展性:

  • 预留扩展空间:在设计协议时,应考虑到未来的扩展需求,通过预留字段或功能接口,使得在不改变协议基本结构的前提下能够增加新的功能或修改现有功能。
  • 分层设计:将通信协议划分为多个层次,每个层次只负责特定的功能,有助于实现功能的独立性和可扩展性。

3) 互操作性:确保不同系统和硬件间的通信,协议应设计为能够在不同的系统和硬件之间进行通信,以实现广泛的互操作性。这有助于不同设备或系统之间的无缝集成。

4) 效率:减少通信开销,协议应尽可能减少通信过程中的带宽使用、处理时间和电力消耗。这可以通过优化协议格式、减少冗余数据等方式实现。

5) 可靠性:提供错误检测和恢复机制,协议应具备错误检测和恢复机制,以确保数据的正确传输。这包括校验码、重传机制等,以应对通信过程中的错误和丢包问题。

6) 安全性:提供适当的安全机制,协议应提供加密、身份验证等安全机制,以防止未经授权的访问和数据篡改。这有助于保护通信过程中的数据安全性和隐私性。

7) 兼容性考虑:  向后兼容性,在设计新版本的协议时,应考虑到与旧版本的兼容性,以确保旧设备或系统能够继续与新设备进行通信。

2.注意事项

2.1.大小端编码问题

        不同的平台存在大小端的问题(即主机字节序和网络字节序),在设计协议格式时,如果协议中存在整型字段,建议使用同一个字节序。通常的做法是在进行网络传输时将所有的整型转换为网络字节序(大端编码,Big Endian),避免不同的机器因为大小端问题解析得到不同的整型值。

        当然,不一定非要转换为网络字节序,如果明确的知道通信的双方使用的是相同的字节序,则也可以不转换。

2.2.字节对齐

以C++为例,通信协议一般用结构体struct来表达,如:

#pragma pack(push, 1)//传输的整个报文
typedef  struct  _stShortWavePacket
{quint32        version; //版本号quint16        frameIndex;quint16        srcObjMark; //源对象标识quint16        srcObjNO; //源对象编号quint16        destObjMark; //目的对象标识quint16        destObjNO; //目的对象编号quint8         isMustResponse; //是否需要应答quint8		   responseStatus; //0:返回成功;非0:失败qint8          content[1024]; //协议内容
}stShortWavePacket;#pragma pack()

        有一组成对的 #pragma XX 指令,其中 #pragma pack(push, n),是告诉编译器接下来的所有结构体(这里就是 userinfo 协议)的每一个字段按 n 个字节对齐,这里 n = 1,按一个字节对齐,即去除任何 padding 字节。这样做的目的是为了内存更加紧凑,节省存储空间。

        我们不再需要这个对齐功能后,应该使用 #pragma pack(pop) 让编译器恢复之前的对齐方式。

注意:#pragma pack(push, n) 与 #pragma pack(pop) 一定要成对使用,如果你漏掉其中任何一个,编译出来的代码可能会出现很多奇怪的运行结果。

2.3.显式指定整型字段的长度

        对于一个 int 型字段,在作为协议传输时,我们应该显式地指定该类型的长度,也就是说,你应该使用 int32_t、int64_t 这样的类型来代替 int、long。之所以这么做,是因为在不同字长的机器上,对于默认的 int 和 long 的长度可能不一样,例如 long 型,在 32 位操作系统上其长度是 4 个字节,而在 64 位机器上其长度是 8 个字节。如果不显式指定这种整形的长度,可能因为不同机器字长不同,导致协议解析出错或者产生错误的结果。

2.4.涉及到浮点数要考虑精度问题,建议放大成整数或者使用字符串去传输

        由于计算机表示浮点数存在精度取舍不准确的问题,例如对于 1.000000,有的计算机可能会得到 0.999999,在某些应用中,如果这个浮点数的业务单位比较大(如表示金额,单位为亿),就会造成很大的影响。因此为了避免不同的机器解析得到不同的结果,建议在网络传输时将浮点数值放大相应的倍数变成整数或者转换为字符串来进行传输。

2.5.协议与自动升级功能

        对于一个商业的产品,发布出去的客户端一般通过客户端的自动升级功能去获得更新(IOS App 除外,苹果公司要求所有的 App 必须在其 App Store 上更新新版本,禁止热更新)。在客户端与服务器通信的所有协议格式中,自动升级协议是最重要的一个,无论版本如何迭代,一定要保证自动升级协议的新旧兼容,这样做有如下原因:

  • 如果新的服务器不能兼容旧客户端中的自动升级协议,那么旧的客户端用户将无法升级成新的版本了,这样的产品相当于把自己给“阉割”了。对于不少产品,不通过自动升级而让众多用户去官网下载新的版本是一件很难做到的事情,这种决策可能会导致大量用户流失;

  • 退一步讲,对于一些测试不完善,或者处于快速迭代中的产品,只要保证自动升级功能正常,旧版本任何 bug 和瑕疵都可以通过升级新版本解决。这对于一些想投放市场试水,但又可能设计不充分的产品尤其重要。

顺便提一下,一般自动升级功能是根据当前版本的版本号与服务器端新版本的版本号进行比较,如果二者之间存在一个大版本号的差别(如1.0.0 与 2.0.0),即有重大功能更新,则应该强制客户端更新下载最新版本;如果只是一个小版本号的更新(如 1.0.0 与 1.1.0),则可以让用户选择是否更新。当然,如果是新版本修正了前一个版本中严重影响使用的 bug,也应当强制用户更新。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 线性约束最小方差准则(LCMV)波束形成算法及MATLAB深入仿真分析
  • 9.1写论文
  • JavaSE-递归法解决二分查找、快速排序
  • Qt:玩转QPainter后转之太极图
  • 蓝色炫酷碎粒子HTML5导航源码
  • go to + 名词 - go to the + 名词
  • shell了解和问答机制
  • C#通过OpenCL调用显卡GPU做高效并行运算
  • c++ 原型模式
  • tornado获取请求参数
  • Qt中的父窗口子窗口和父类子类的区别
  • Nginx: 负载均衡基础配置, 加权轮序, hash算法, ip_hash算法, least_conn算法
  • 单一职责原则介绍
  • wordpress 页面URL自动跳转到图片地址?
  • H5手机端调起支付宝app支付
  • 《Javascript数据结构和算法》笔记-「字典和散列表」
  • ES6核心特性
  • ES学习笔记(10)--ES6中的函数和数组补漏
  • JS进阶 - JS 、JS-Web-API与DOM、BOM
  • nginx(二):进阶配置介绍--rewrite用法,压缩,https虚拟主机等
  • Object.assign方法不能实现深复制
  • React 快速上手 - 06 容器组件、展示组件、操作组件
  • SegmentFault 社区上线小程序开发频道,助力小程序开发者生态
  • spring cloud gateway 源码解析(4)跨域问题处理
  • vue的全局变量和全局拦截请求器
  • 从setTimeout-setInterval看JS线程
  • 前端 CSS : 5# 纯 CSS 实现24小时超市
  • 前言-如何学习区块链
  • 如何使用Mybatis第三方插件--PageHelper实现分页操作
  • 如何在 Tornado 中实现 Middleware
  • 数据库写操作弃用“SELECT ... FOR UPDATE”解决方案
  • 通过获取异步加载JS文件进度实现一个canvas环形loading图
  • 学习Vue.js的五个小例子
  • 一道面试题引发的“血案”
  • 在Mac OS X上安装 Ruby运行环境
  • 中国人寿如何基于容器搭建金融PaaS云平台
  • Java数据解析之JSON
  • 宾利慕尚创始人典藏版国内首秀,2025年前实现全系车型电动化 | 2019上海车展 ...
  • 带你开发类似Pokemon Go的AR游戏
  • # linux 中使用 visudo 命令,怎么保存退出?
  • #if 1...#endif
  • $con= MySQL有关填空题_2015年计算机二级考试《MySQL》提高练习题(10)
  • $var=htmlencode(“‘);alert(‘2“); 的个人理解
  • (二)JAVA使用POI操作excel
  • (二)换源+apt-get基础配置+搜狗拼音
  • (附源码)springboot宠物管理系统 毕业设计 121654
  • (简单) HDU 2612 Find a way,BFS。
  • (收藏)Git和Repo扫盲——如何取得Android源代码
  • (学习日记)2024.01.09
  • (一)Dubbo快速入门、介绍、使用
  • (转) SpringBoot:使用spring-boot-devtools进行热部署以及不生效的问题解决
  • (转)Windows2003安全设置/维护
  • ****三次握手和四次挥手
  • .NET 8 中引入新的 IHostedLifecycleService 接口 实现定时任务
  • .net core 调用c dll_用C++生成一个简单的DLL文件VS2008