使用C语言自定义序列化与反序列化,快速实现自定义协议,二进制传输,类似protobuf
//编程网站查看
// /c/#id/c85fa97f81596701975d3ae7b9346725
//位数一定要和平台相同
typedef unsigned int u32;
typedef unsigned short u16;
typedef unsigned char u8;
#include ""
#include ""
#include ""
#define LOCK1 'D'
#define LOCK2 'E'
#define R_CODE 0XF1//异或密码
#define RE_SEND_BUFFER 22
//协议1字节对齐
typedef struct{
u8 MsgHead[2];//头字节
u8 MsgLength;//有效数据长度
u8 MsgClass;//消息类
u16 MsgCrc16;//校验
u8 MsgPayLoad[RE_SEND_BUFFER];//消息载荷
}MsgProtocol_t;
//格式化
typedef struct
{
void (*FmtU8)(u8 d);
void (*FmtU16)(u16 d);
void (*FmtU32)(u32 d);
void (*FmtStr)(char* str);
void (*FmtHex)(u8* Dt,u16 len);
void (*FmtHead)(void);
void (*Fmtend)(void);
void (*FmtCmd)(u16 cmd);
void (*FmtCrc16)(void);
u8* (*GetBuffer)(void);
u16 (*GetLength)(void);
}ProtocolFormat_t;
MsgProtocol_t transferProtocol;
ProtocolFormat_t* Vest_TxRxProtocol = NULL;
static u32 SendProtocolLength = 0;
//CRC16高
const u8 CRC_Array_H[] =
{
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40
};
//CRC16低
const u8 CRC_Array_L[] =
{
0x00, 0xC0, 0xC1, 0x01, 0xC3, 0x03, 0x02, 0xC2, 0xC6, 0x06,
0x07, 0xC7, 0x05, 0xC5, 0xC4, 0x04, 0xCC, 0x0C, 0x0D, 0xCD,
0x0F, 0xCF, 0xCE, 0x0E, 0x0A, 0xCA, 0xCB, 0x0B, 0xC9, 0x09,
0x08, 0xC8, 0xD8, 0x18, 0x19, 0xD9, 0x1B, 0xDB, 0xDA, 0x1A,
0x1E, 0xDE, 0xDF, 0x1F, 0xDD, 0x1D, 0x1C, 0xDC, 0x14, 0xD4,
0xD5, 0x15, 0xD7, 0x17, 0x16, 0xD6, 0xD2, 0x12, 0x13, 0xD3,
0x11, 0xD1, 0xD0, 0x10, 0xF0, 0x30, 0x31, 0xF1, 0x33, 0xF3,
0xF2, 0x32, 0x36, 0xF6, 0xF7, 0x37, 0xF5, 0x35, 0x34, 0xF4,
0x3C, 0xFC, 0xFD, 0x3D, 0xFF, 0x3F, 0x3E, 0xFE, 0xFA, 0x3A,
0x3B, 0xFB, 0x39, 0xF9, 0xF8, 0x38, 0x28, 0xE8, 0xE9, 0x29,
0xEB, 0x2B, 0x2A, 0xEA, 0xEE, 0x2E, 0x2F, 0xEF, 0x2D, 0xED,
0xEC, 0x2C, 0xE4, 0x24, 0x25, 0xE5, 0x27, 0xE7, 0xE6, 0x26,
0x22, 0xE2, 0xE3, 0x23, 0xE1, 0x21, 0x20, 0xE0, 0xA0, 0x60,
0x61, 0xA1, 0x63, 0xA3, 0xA2, 0x62, 0x66, 0xA6, 0xA7, 0x67,
0xA5, 0x65, 0x64, 0xA4, 0x6C, 0xAC, 0xAD, 0x6D, 0xAF, 0x6F,
0x6E, 0xAE, 0xAA, 0x6A, 0x6B, 0xAB, 0x69, 0xA9, 0xA8, 0x68,
0x78, 0xB8, 0xB9, 0x79, 0xBB, 0x7B, 0x7A, 0xBA, 0xBE, 0x7E,
0x7F, 0xBF, 0x7D, 0xBD, 0xBC, 0x7C, 0xB4, 0x74, 0x75, 0xB5,
0x77, 0xB7, 0xB6, 0x76, 0x72, 0xB2, 0xB3, 0x73, 0xB1, 0x71,
0x70, 0xB0, 0x50, 0x90, 0x91, 0x51, 0x93, 0x53, 0x52, 0x92,
0x96, 0x56, 0x57, 0x97, 0x55, 0x95, 0x94, 0x54, 0x9C, 0x5C,
0x5D, 0x9D, 0x5F, 0x9F, 0x9E, 0x5E, 0x5A, 0x9A, 0x9B, 0x5B,
0x99, 0x59, 0x58, 0x98, 0x88, 0x48, 0x49, 0x89, 0x4B, 0x8B,
0x8A, 0x4A, 0x4E, 0x8E, 0x8F, 0x4F, 0x8D, 0x4D, 0x4C, 0x8C,
0x44, 0x84, 0x85, 0x45, 0x87, 0x47, 0x46, 0x86, 0x82, 0x42,
0x43, 0x83, 0x41, 0x81, 0x80, 0x40
};
u16 Message_count_CRC_2(u8 CrcLenth,u8 *CRC_Data)
{
u8 Crc_H = 0xFF;
u8 Crc_L = 0xFF;
u8 Index;
while (CrcLenth--)
{
Index = Crc_H ^ *CRC_Data++;
Crc_H = Crc_L ^ CRC_Array_H[Index];
Crc_L = CRC_Array_L[Index];
}
return((Crc_H << 8) | Crc_L);
}
//8
static void ThisFmtU8(u8 d)
{
transferProtocol.MsgPayLoad[SendProtocolLength++] = d;
}
//16
static void ThisFmtU16(u16 d)
{
memcpy(&transferProtocol.MsgPayLoad[SendProtocolLength],&d,2);
SendProtocolLength += 2;
}
//32
static void ThisFmtU32(u32 d)
{
memcpy(&transferProtocol.MsgPayLoad[SendProtocolLength],&d,4);
SendProtocolLength += 4;
}
//str
static void ThisFmtStr(char* str)
{
memcpy(&transferProtocol.MsgPayLoad[SendProtocolLength],str,strlen(str));
SendProtocolLength += strlen(str);
}
//cmd
static void ThisFmtCmd(u16 cmd)
{
u16 tmp_cmd = cmd ^ R_CODE;
transferProtocol.MsgClass = tmp_cmd;
}
//heard
static void ThisFmtHead(void)
{
SendProtocolLength = 0;
//头
transferProtocol.MsgHead[0] = LOCK1 ^ R_CODE;
transferProtocol.MsgHead[1] = LOCK2 ^ R_CODE;
}
//crc16
static void ThisFmtCrc16(void)
{
transferProtocol.MsgCrc16 = Message_count_CRC_2(SendProtocolLength,transferProtocol.MsgPayLoad)^R_CODE;
}
//end
static void ThisFmtend(void)
{
//总数居长度
transferProtocol.MsgLength = (SendProtocolLength + (sizeof(MsgProtocol_t) -RE_SEND_BUFFER)) ^R_CODE;
}
//get buffer
static u8* ThisGetBuffer(void)
{
return (u8*)&transferProtocol;
}
//hex
static void ThisFmtHex(u8* Dt,u16 Len)
{
memcpy(&transferProtocol.MsgPayLoad[SendProtocolLength],Dt,Len);
SendProtocolLength += Len;
}
//得到数据包总长度
static u16 ThisGetLength(void)
{
return SendProtocolLength + (sizeof(MsgProtocol_t) -RE_SEND_BUFFER);
}
//get object
ProtocolFormat_t* ProtocolFormatNew(void)
{
//如果使用的C99编译器的话可以直接初始化赋值
//这里使用的是静态代表每次获取实例的对象是一样的
//要想不同可以使用malloc分配内存
static ProtocolFormat_t SendProtocol;
SendProtocol.FmtU8 = ThisFmtU8;
SendProtocol.FmtU16 = ThisFmtU16;
SendProtocol.FmtU32 = ThisFmtU32;
SendProtocol.FmtHex = ThisFmtHex,
SendProtocol.FmtStr = ThisFmtStr;
SendProtocol.FmtHead = ThisFmtHead;
SendProtocol.Fmtend = ThisFmtend;
SendProtocol.FmtCmd = ThisFmtCmd;
SendProtocol.FmtCrc16 = ThisFmtCrc16;
SendProtocol.GetBuffer = ThisGetBuffer;
SendProtocol.GetLength = ThisGetLength;
return &SendProtocol;
}
//接收端对协议分析还原
MsgProtocol_t* Protocol_Analysis(void* d)
{
MsgProtocol_t *p = (MsgProtocol_t*)d;
p->MsgHead[0] = p->MsgHead[0] ^ R_CODE;
p->MsgHead[1] = p->MsgHead[1] ^ R_CODE;
p->MsgClass = p->MsgClass ^ R_CODE;
p->MsgLength = p->MsgLength ^ R_CODE;
p->MsgCrc16 = p->MsgCrc16 ^ R_CODE;
return p;
}
//测试buf
static u8 DisplayBuffer[128];
//序列化
static void TestCTL1(void)
{
u8 i = 0;
Vest_TxRxProtocol->FmtHead();
Vest_TxRxProtocol->FmtCmd(0x61);
Vest_TxRxProtocol->FmtU8(0x01);
Vest_TxRxProtocol->FmtCrc16();
Vest_TxRxProtocol->Fmtend();
memcpy(DisplayBuffer,Vest_TxRxProtocol->GetBuffer(),Vest_TxRxProtocol->GetLength());
for(i=0;i < Vest_TxRxProtocol->GetLength();i++)
printf("%02x ",DisplayBuffer[i]);
printf("\n");
}
//反序列化
//反序列化测试
static u8 DecodeTestBuffer[7] = {0xb5,0xb4,0xf6,0x90,0x71,0x7e,0x01};
static void DecodeTestHandler(void* p,u16 length)
{
u8 *Data = (u8*)p;
u16 indexof = 0;
u8 i =0;
if(length > 6)
{
while( (length-indexof) >= 7)//解包
{
char head[2];
head[0] = Data[indexof] ^ R_CODE;
head[1] = Data[indexof+1] ^ R_CODE;
if(head[0] == LOCK1 && head[1] == LOCK2)
{
MsgProtocol_t *p = Protocol_Analysis(&Data[indexof]);
printf("cmd=0x%02x,crc16=0x%02x,len=0x%02x\n",p->MsgClass,p->MsgCrc16,p->MsgLength);
printf("Data: ");
for(i=0;i<p->MsgLength-6;i++)
{
printf("%02X ",p->MsgPayLoad[i]);
}
indexof += p->MsgLength;//下一包数据
}
else{
indexof++;
}
}
}
}
int main(void)
{
Vest_TxRxProtocol = ProtocolFormatNew();
TestCTL1();
DecodeTestHandler(DecodeTestBuffer,7);
}
//C语言在线测试网站
///c/