自学内容网 自学内容网

C# —— CRC16 算法

CRC16:即循环冗余校验码。数据通信当中一种常用的查错校验码

其特征信息字段和校验字段的长度可以是任意选定的,对数据进行指定多项式计算
并且将得到的结果附加在帧的后面,接受的设备也执行类似的算法,以保证数据传输的正确性和完整性

crc-16过程?

        1 初始化一个16位的寄存器地址 用作初始值
        2 遍历数据字节,从最高位到最低位,
        3 将数据字节与寄存器异或
        4 对寄存器进行8次迭代,每一次迭代将寄存器右移一位
        5 如果最低位位1,将寄存器与生成多项式0x8005异或,否则只进行右移操作
        6 重复上述步骤直到遍历完所有的字节
        7 最终寄存器的值就是crc16校验码
        8 crc计算之后高低位进行互换。

实例:

创建生成按钮

创建封装CRC方法

public static byte[] CRCCalc(byte[] data)  //[]
{
    //1 初始化一个16位的寄存器地址 用作初始值
    int crc = 0xffff;

    //2 遍历数据字节
    for(int i = 0; i < data.Length; i++)
    {
        //3将数据字节与寄存器异或
        crc = crc ^ data[i];
        //4对寄存器进行8次迭代,每一次迭代将寄存器右移一位
        for(int j = 0; j < 8; j++)
        {
            int temp;
            temp = crc & 1;
            crc = crc >> 1;//每一次迭代将寄存器右移一位
            crc = crc & 0x7fff;
            //5 如果最低位位1,将寄存器与生成多项式0x8005异或,否则只进行右移操作
            if (temp == 1)
            {
                crc = crc ^ 0xa001;
            }
            crc = crc & 0xffff;

        }
    }
    byte[] crc16 = new byte[2];// crc寄存器的高低为进行互换
    crc16[1] =(byte)((crc>>8) & 0xff); //crc寄存器高八位变成了八低位
    crc16[0] = (byte)(crc & 0xff);// crc寄存器低八位变成了高低位
    return crc16;
}

/// <summary>
/// CRC校验,参数为空格或逗号间隔的字符串
/// </summary>
/// <param name="data">校验数据,逗号或空格间隔的16进制字符串(带有0x或0X也可以),逗号与空格不能混用</param>
/// <returns>字节0是高8位,字节1是低8位</returns>
///"01 03 00 03 00 01"
public static byte[] CRCCalc(string data)
{
    //分隔符是空格还是逗号进行分类,并去除输入字符串中的多余空格
    IEnumerable<string> datac = data.Contains(",") ? data.Replace(" ", "").Replace("0x", "").Replace("0X", "").Trim().Split(',') : data.Replace("0x", "").Replace("0X", "").Split(' ').ToList().Where(u => u != "");
    List<byte> bytedata = new List<byte>();
    foreach (string str in datac)
    {
        bytedata.Add(byte.Parse(str, System.Globalization.NumberStyles.AllowHexSpecifier));
    }
    byte[] crcbuf = bytedata.ToArray();
    //crc计算赋初始值
    return CRCCalc(crcbuf);
}

/// <summary>
///  CRC校验,截取data中的一段进行CRC16校验
/// </summary>
/// <param name="data">校验数据,字节数组</param>
/// <param name="offset">从头开始偏移几个byte</param>
/// <param name="length">偏移后取几个字节byte</param>
/// <returns>字节0是高8位,字节1是低8位</returns>
public static byte[] CRCCalc(byte[] data, int offset, int length)
{
    byte[] Tdata = data.Skip(offset).Take(length).ToArray();
    return CRCCalc(Tdata);
}

验证校验码方法是否正确

private void button1_Click(object sender, EventArgs e)
{
    // 1 传递的是字节数组类型
    byte[] data = new byte[] { 0x01, 0x03, 0x00, 0x03, 0x00, 0x01 };
    byte[] crc =  CRCCalc16.CRCCalc(data);
    for (int i = 0;i < crc.Length; i++)
    {
        Console.WriteLine(crc[i]); //十进制116 10  //16进制是74 0A
    }

    //2传递的字符串类型 
   byte[] crc1=   CRCCalc16.CRCCalc("01 03 00 03 00 01");
    for (int i = 0; i < crc1.Length; i++)
    {
        Console.WriteLine(crc1[i]); //十进制116 10  //16进制是74 0A
    }

    //3传递是字节数组和 字节开始位置和结束位置
    byte[] crc2 = CRCCalc16.CRCCalc(data,0,data.Length);
    for (int i = 0; i < crc2.Length; i++)
    {
        Console.WriteLine(crc2[i]); //十进制116 10  //16进制是74 0A
    }
}


原文地址:https://blog.csdn.net/lu2023_8_6/article/details/140476798

免责声明:本站文章内容转载自网络资源,如本站内容侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!