本文共 1599 字,大约阅读时间需要 5 分钟。
让我们从基础开始,逐步分析并解决问题。
CRC(循环冗余校验)是一种用于检测数据传输或存储中是否发生错误的简单且有效的方法。它通过使用一个预先确定的多项式,对数据进行计算,并生成一个校验值,与实际数据一起传输。接收端可以使用相同的多项式对数据和校验值进行计算,比较结果是否一致,从而判断数据是否完整无误。
在本文中,我们使用的是 CRC-16 校验,多项式为 0xA001。
假设我们有以下数据(十六进制):
01 03 61 00 00 02
这个数据总共有6个字节,换算成二进制则为:
01 03 61 00 00 02 -> 00000001 00000011 01100001 00000000 00000000 00000010
将这些二进制数据按顺序连接起来,得到一个16位的二进制数:
00000001 00000011 01100001 00000000 00000000 00000010 -> 00000001 00000011 01100001 00000000 00000000 00000010
将其转换为十六进制:
01 03 61 00 00 02
接收端会使用多项式 0xA001 进行计算,生成一个校验值。校验值的计算过程是对数据与多项式进行按位异或运算。
计算过程如下:
0xffff(即 11111111 11111111)。0xA001(即 00000010 00000001)进行异或运算。让我们用代码来模拟这一过程:
unsigned short tmp = 0xffff; // 初始化寄存器为 0xffffunsigned short ret1 = 0; // 用于存储校验值for(int n = 0; n < 6; n++){ tmp = buff[n] ^ tmp;}for(int i = 0; i < 8; i++){ if(tmp & 0x01){ tmp = tmp >> 1; tmp = tmp ^ 0xa001; } else { tmp = tmp >> 1; }} 按照上述代码,输入数据 01 03 61 00 00 02,计算得到校验值为 F7DB。
用户提到“校验值的高低位顺序反转”,即 F7DB 反转后变为 DBF7。这在实际应用中可能会导致校验失败,因为校验值的高低位顺序必须与传输数据的高低位顺序一致。
在实际应用中,校验值的高低位顺序应与数据的高低位顺序保持一致。因此,如果数据的高低位顺序是 高 -> 低,校验值的高低位顺序也应保持 高 -> 低。如果需要改变顺序,可以在计算校验值后进行反转处理。
在代码中,我们可以直接将 tmp 变量进行反转处理:
ret1 = tmp >> 8; // 提取校验值的高 8 位ret1 |= (tmp << 8); // 将低 8 位移到高位
这样,ret1 就变成了反转后的校验值。
经过计算,校验值为 F7DB。反转后为 DBF7。
在实际应用中,必须确保校验值的高低位顺序与数据的高低位顺序保持一致。如果需要改变顺序,应在计算校验值后进行反转处理。通过上述步骤,我们可以正确地生成和验证 CRC 校验值,确保数据传输的完整性和可靠性。
转载地址:http://lcpb.baihongyu.com/