接触过网络协议的人对TLV一定或多或少的知道.作为一种自定义应用层标准.

TLV使用十分广泛.他对数据封包有着很好的定义,简单实用.

TLV即Type-Length-Value.即我们每个封装成TLV包的数据都必须为其添加Type和Length字段

TLV示意图如下:


大家首先要区分数据包和数据报.本文的实例仅仅针对TLV数据包.

而并未添加注册一些控制信令和报头形成数据报,而其中实际的数据

有TLV包组成.

TLV数据包和数据报的关系可由下图表示:

此外,TLV本身有两种结构,一种是基本TLV结构,另外一种是嵌套TLV结构.而本文使用的是嵌套TLV结构.

基本TLV包:


嵌套TLV包:



本文使用嵌套TLV包.



几点说明:

编码方法

1. 将类型type用htonl转换为网络字节顺序,指针偏移+4

2. 将长度length用htonl转换为网络字节顺序,指针偏移+4

3. 若值value数据类型为int、char、short,则将其转换为网络字节顺序,指针偏移+4;若值为字符串类型,写进后,指针偏移+length

解码方法

1. 读取type 用ntohl转换为主机字节序得到类型,指针偏移+4

2. 读取lengh用ntohl转换为主机字节序得到长度;指针偏移+4

3. 根据得到的长度读取value,若value数据类型为int、char、short,用ntohl转换为主机字节序,指针偏移+4;若value数据类型为字符串类型,指针偏移+length

Type和Length的长度固定,一般那是2、4个字节(这里统一采用4个字节);

Value的长度有Length指定



Demo代码如下:

 #include <stdio.h>
#include <WinSock2.h>
#include <string> #pragma comment(lib, "WS2_32") //定义枚举类型常量,来填充Tpye字段,其中emTlvNRoot填充根TLV包的Type
//emTlvName字段用于填充子TLV字段中名字的Type字段.emTlvAge,emTlvColor类似
//此类型字段是为了TLV包解码时识别到底是哪个TLV包.进而解析出对应的数据.
enum emTLVNodeType
{
emTlvNNone = ,
emTlvNRoot, //根节点
emTlvName, //名字
emTlvAge, //年龄
emTlvColor //颜色 1 白色 2 黑色
}; //定义要封装成TLV包的数据,包括名字,年龄,颜色。
typedef struct _CAT_INFO
{
char szName[];
int iAge;
int iColor;
}CAT_INFO,*LPCAT_INFO; //此类为TLC类,其中有四个成员函数,WriteInt和Write是用于
//把原始数据封装为TLV包然后存入内存区块.即TLV包编码过程
//而ReadInt和Read用于把内存区块的TLV包解析出来.即为TLV包的解码过程
class CTlvPacket
{ public: CTlvPacket(char *pBuf,unsigned int len):
m_pData(pBuf),m_uiLength(len),m_pEndData(m_pData+len),
m_pWritePtr(m_pData),m_pReadPtr(m_pData) { } ~CTlvPacket() { } bool WriteInt(int data,bool bMovePtr = true)
{
int tmp = htonl(data);
return Write(&tmp,sizeof(int));
} bool Write(const void *pDst,unsigned int uiCount)
{
::memcpy(m_pWritePtr,pDst,uiCount);
m_pWritePtr += uiCount;
return m_pWritePtr < m_pEndData ? true : false;
} bool ReadInt(int *data,bool bMovePtr = true)
{
Read(data,sizeof(int));
*data = ntohl(*data);
return true;
} bool Read(void *pDst,unsigned int uiCount)
{
::memcpy(pDst,m_pReadPtr,uiCount);
m_pReadPtr += uiCount;
return m_pReadPtr < m_pEndData ? true : false;
} private: char *m_pData; unsigned int m_uiLength; char *m_pEndData; char *m_pWritePtr; char *m_pReadPtr; }; /* 格式:
root L1 V
T L V T L V T L V L1 的长度即为“T L V T L V T L V”的长度 */ //此函数实现TLV编码过程
int TLV_EncodeCat(LPCAT_INFO pCatInfo, char *pBuf, int &iLen)
{ if (!pCatInfo || !pBuf)
{
return -;
} CTlvPacket enc(pBuf,iLen);
enc.WriteInt(emTlvNRoot);
enc.WriteInt(++); //length enc.WriteInt(emTlvName);
enc.WriteInt();
enc.Write(pCatInfo->szName,); enc.WriteInt(emTlvAge);
enc.WriteInt();
enc.WriteInt(pCatInfo->iAge); enc.WriteInt(emTlvColor);
enc.WriteInt();
enc.WriteInt(pCatInfo->iColor); iLen = +++; return ; } //此函数实现TLV解码过程
int TLV_DecodeCat(char *pBuf, int iLen, LPCAT_INFO pCatInfo)
{ if (!pCatInfo || !pBuf)
{
return -;
} CTlvPacket encDec(pBuf,iLen);
int iType;
int iSum,iLength; encDec.ReadInt(&iType);
if (emTlvNRoot != iType)
{
return -;
}
encDec.ReadInt(&iSum); //通过判断Type头字段对TLV包进行解析
while (iSum > )
{ encDec.ReadInt(&iType);//读取主TLV包的type头
encDec.ReadInt(&iLength);//读取主TLV包的length头 switch(iType) //此时buff指针移动到子TLV包.并解析子TLV的type头字段
{ case emTlvName:
encDec.Read(pCatInfo->szName,);
iSum -= ;
break; case emTlvAge:
encDec.ReadInt(&pCatInfo->iAge);
iSum -= ;
break; case emTlvColor:
encDec.ReadInt(&pCatInfo->iColor);
iSum -= ;
break; default:
printf("TLV_DecodeCat unkonwn error. \n");
break; } } return ; } //主函数
int main(int argc, char* argv[])
{ int iRet, iLen;
char buf[] = {}; CAT_INFO cat; //cat为定义的原始数据包括name,age,color
memset(&cat,,sizeof(cat));//cat结构体初始化 //对cat对象赋值
strcpy(cat.szName,"Tom");
cat.iAge = ;
cat.iColor = ; //实现对cat对象的编码,编码结果存储在buf中.
iRet = TLV_EncodeCat(&cat,buf,iLen); //TLV编码成功与否的判断
if ( == iRet )
{
printf("TLV_EncodeCat ok, iLen = %d. \n",iLen);
}
else
{
printf("TLV_EncodeCat error \n");
} //将cat结构置为0
memset(&cat,,sizeof(cat)); //TLV包解码过程,将解包后的数据存入cat结构体对象
iRet = TLV_DecodeCat(buf,iLen,&cat); //输出解包后的结构体数据
if ( == iRet )
{
printf("TLV_DecodeCat ok, cat name = %s, age = %d, color = %d. \n",cat.szName,cat.iAge,cat.iColor);
}
else
{
printf("TLV_DecodeCat error, code = %d. \n", iRet);
} int iWait = getchar(); return ;
}

运行结果截图如下:

这里要对TLV包长:iLen = 8+20+12+12; 进行说明.

为什么是这样呢.

因为一个完整的TLV包的主包包含Type字段和Length字段,每个字段占用4个字节所以共八个字节.

而主TLV包的Value字段包含三个子TLV包.

第一个子TLV包为name,而char szName[12],加之子包Type和子包Length,所以一共20个字节

同理,对于age子包共计八个字节,color子包共计八个字节.

所以整个TLV包的长度就为8+20+12+12



本文参考自博客:http://blog.csdn.net/chexlong/article/details/6974201

 

TLV----Demo讲解的更多相关文章

  1. Swift轻松入门——基本语法介绍和详细地Demo讲解(利用WebView打开百度、新浪等网页)

    转载请务必注明出处(all copyright reserved by iOSGeek) 本文主要分为两个部分,第一部分介绍Swift的基本语法,第二部分讲解一个利用WebView来打开百度.sina ...

  2. 02——微信小程序官方demo讲解——app部分

    第一节讲了目录结构,这节主要讲解下目录中app.js部分. 它由三部分组成app.js.app.json与app.wxss 1.JS部分 1.1概述 //app.js App({ onLaunch: ...

  3. C# webApi 与 AngularJs 实现增删改Demo 讲解(一)

    公司在使用webAPI+AngularJs+SlcikGrid进行产品开发,自己也是初学Angular,就做了一个Demo,实现增删改功能,希望可以帮助大家. 界面如同所示:  数据库一张单表很简单, ...

  4. 支付宝即时到账接口开发 - DEMO讲解

    支付宝即时到帐接口 环境要求 PHP5.0以上,且需要开启curl.openssl. 文档地址: https://doc.open.alipay.com/doc2/detail?treeId=62&a ...

  5. 03——微信小程序官方demo讲解——page部分

    一个page由一个文件夹以及文件夹下四个文件组成. 比如一个页面叫index.则需要在pages目录下新建一个index目录,且包含由index+类型(js\wxml\wxss\json)为名组成的若 ...

  6. 01——微信小程序官方demo讲解——文件结构

    1.环境概览 首先环境配置的部分略过,打开小程序开发工具.选择一个空目录,即可开始一个demo项目. 其中新建成功后的目录如图所示: 2.文件结构描述 如图所示,左边是界面展示,右边是目录结构. 目录 ...

  7. WCF开发的流程-服务端和客户端之间的通讯(内含demo讲解)

    讲解技术之前,恳请博友让我说几句废话.今天是我第一在博客园发布属于自己原创的博文(如有雷同,那是绝对不可能的事,嘿嘿).之前一直是拜读各位博友的大作,受益匪浅的我在这对博友们说声谢谢,谢谢你们的共享! ...

  8. 分享-结合demo讲解JS引擎工作原理

    代码如下: var x = 1; function A(y){ var x = 2; function B(z){ console.log(x+y+z); } return B; } var C = ...

  9. 【SSH系列】初识spring+入门demo

    学习过了hibernate,也就是冬天,经过一个冬天的冬眠,当春风吹绿大地,万物复苏,我们迎来了spring,在前面的一系列博文中,小编介绍hibernate的相关知识,接下来的博文中,小编将继续介绍 ...

  10. RabbitMQ的使用Demo

    rabbitmq消息队列,官网有六种,实战常用的也就如下五种. 下面开始demo讲解 大致三步:1.配置消息队列,2.生产者提供消息给队列,3.消费者监听消费队列消息 源码下载:https://pan ...

随机推荐

  1. SqlDataAdapter类

    SqlDataAdapter类常用操作 SqlDataAdapter常用于从数据库中返回一个结果集时. 常用操作: Fill(); 示例: static void Main(string[] args ...

  2. python实现词法分析

    #请先安装Ply # -*- coding: utf-8 -*- #------------------------------------------------------------------ ...

  3. Implement Trie (Prefix Tree) 解答

    Question Implement a trie with insert, search, and startsWith methods. Note:You may assume that all ...

  4. uva 101 by sixleaves

    这是一道很好的模拟题,用vector<int> p[maxn],建立模型,映射为maxn个堆.主要要掌握vector模拟堆操作的简单方法.接下来得思路是自顶向下的方式,逐步完善程序.首先根 ...

  5. 2013长沙网络赛H题Hypersphere (蛋疼的题目 神似邀请赛A题)

    Hypersphere Time Limit: 1 Second       Memory Limit: 32768 KB In the world of k-dimension, there's a ...

  6. Elasticsearch 安装与集群配置

    一.软件版本 操作系统:CentOS-6.5-x86_64 ES版本:5.0 主机:192.168.63.246 主机: 192.168.63.242 二.部署环境规划:   1. 需求:jdk版本: ...

  7. python3 时间和日期

    Python程序可以通过多种方式来处理日期和时间.日期格式之间的转换是计算机的一个共同核心.Python的时间和日历模块能够帮助我们跟踪的日期和时间. 什么是刻度? 时间间隔以秒为单位的浮点数.特别是 ...

  8. Pasha and String(思维,技巧)

    Pasha and String Time Limit:2000MS     Memory Limit:262144KB     64bit IO Format:%I64d & %I64u S ...

  9. 自定义视图控制器切换(iOS)

    在iOS开发过程中,通常我们会使用UINavigationController,UITabbarController等苹果提供的视图控制器来切换我们的视图.在iOS5之前,如果要自定义容器视图控制器很 ...

  10. 正式软件工作第一天————MVC、ext JS、和clsa

    正式上班第一天,引导人就提出了MVC.extjs和clsa 三种框架,在这之前只听说了MVC,但还木有学习过,顺便把学习过程记录下来. MVC——初学习 1.M——model 模型:应用对象,处理业务 ...