随笔 - 298  文章 - 10  评论 - 87  2

CRC(循环冗余校验)在线计算 http://www.ip33.com/crc.html

里面的8005的多项式值,但网上看到的算法都是用A001来异或的

-----------------------------------------------------

0x8005=1000 0000 0000 0101B 

0xA001=1010 0000 0000 0001B

对比两个二进制高低位正好是完全相反的,CRC校验分为正向校验与反向校验。正向校验高位在左,反向校验低位在左

正向校验使用左移位,反向校验使用右移位

---------------------------------

Modbus协议,常规485通讯的信息发送形式如下:

地址 功能码 数据信息 校验码

1byte 1byte nbyte 2byte

CRC校验是前面几段数据内容的校验值,为一个16位数据,发送时,低8位在前,高8为最后。

例如:信息字段代码为: 1011001,校验字段为:1010。

发送方:发出的传输字段为: 1 0 1 1 0 0 1 1 0 10

 

参考文档:

 https://www.jianshu.com/p/676744381473

https://www.jianshu.com/p/c0d93c2e89ce

VB对应的算法

Public Function CRC16(Data() As Byte, ByRef bLow As Byte, ByRef bHigh As Byte) As String

    Dim CRC16Lo As Byte, CRC16Hi As Byte      'CRC寄存器
    Dim CL As Byte, ch As Byte                '多项式码&HA001
    Dim SaveHi As Byte, SaveLo As Byte
    Dim i As Integer
    Dim flag As Integer
    CRC16Lo = &HFF
    CRC16Hi = &HFF
    CL = 1
    ch = &HA0
    
    For i = LBound(Data) To UBound(Data) - 2
        CRC16Lo = CRC16Lo Xor Data(i) '每一个数据与CRC寄存器的低8位进行异或
        '右移8次
        For flag = 0 To 7
            '记录一下,用来判断移出位是0还是1
            SaveHi = CRC16Hi
            SaveLo = CRC16Lo
            
            CRC16Hi = CRC16Hi \ 2            '高位右移一位
            CRC16Lo = CRC16Lo \ 2            '低位右移一位
            
            If ((SaveHi And &H1) = &H1) Then '如果高位字节最后一位为1
                CRC16Lo = CRC16Lo Or &H80      '则低位字节右移后前面补1
            End If                           '否则自动补0
            
            If ((SaveLo And &H1) = &H1) Then '如果移出位为1,则与多项式码进行异或
                CRC16Hi = CRC16Hi Xor ch
                CRC16Lo = CRC16Lo Xor CL
            End If
        Next flag
    Next i
    Dim ReturnData(1) As Byte
    ReturnData(0) = CRC16Hi              'CRC高位
    ReturnData(1) = CRC16Lo              'CRC低位
    Dim Value As Double
    Value = CLng(CRC16Hi * 256) + CRC16Lo
    CRC16 = ReturnData
    bHigh = CRC16Hi
    bLow = CRC16Lo


End Function

 

C# 对应的算法

        public static byte[] CRC16(byte[] data)
        {
            int len = data.Length;
            if (len > 0)
            {
                ushort crc = 0xFFFF;

                for (int i = 0; i < len; i++)
                {
                    crc = (ushort)(crc ^ (data[i]));
                    for (int j = 0; j < 8; j++)
                    {
                        crc = (crc & 1) != 0 ? (ushort)((crc >> 1) ^ 0xA001) : (ushort)(crc >> 1);
                    }
                    System.Diagnostics.Debug.WriteLine("i=" + i.ToString() + ","+ crc.ToString());
                }
                byte hi = (byte)((crc & 0xFF00) >> 8); //高位置
                byte lo = (byte)(crc & 0x00FF); //低位置

                return new byte[] { hi, lo };
            }
            return new byte[] { 0, 0 };
        }

 

posted on 2019-10-22 23:16  Gu  阅读(...)  评论(...编辑  收藏