CRC16算法

/**
 * @author yxl
 * @date 2022-04-25
 *
 * CRC16校验码计算
 * <p>
 * (1).预置1个16位的寄存器为十六进制FFFF(即全为1),称此寄存器为CRC寄存器;
 * (2).把第一个8位二进制数据(既通讯信息帧的第一个字节)与16位的CRC寄存器的低
 * 8位相异或,把结果放于CRC寄存器;
 * (3).把CRC寄存器的内容右移一位(朝低位)用0填补最高位,并检查右移后的移出位;
 * (4).如果移出位为0:重复第3步(再次右移一位);如果移出位为1:CRC寄存器与多项式A001(1010 0000 0000 0001)进行异或;
 * (5).重复步骤3和4,直到右移8次,这样整个8位数据全部进行了处理;
 * (6).重复步骤2到步骤5,进行通讯信息帧下一个字节的处理;
 * (7).将该通讯信息帧所有字节按上述步骤计算完成后,得到的16位CRC寄存器的高、低
 * 字节进行交换;
 * (8).最后得到的CRC寄存器内容即为CRC16码。(注意得到的CRC码即为低前高后顺序)
 */
public class CRC16 {
    /**
     * 计算CRC16校验码
     *
     * @param data 需要校验的字符串
     * @return 校验码
     */
    public static String getCRC(String data) {
        data = data.replace(" ", "");
        int len = data.length();
        if (!(len % 2 == 0)) {
            return "0000";
        }
        int num = len / 2;
        byte[] para = new byte[num];
        for (int i = 0; i < num; i++) {
            int value = Integer.valueOf(data.substring(i * 2, 2 * (i + 1)), 16);
            para[i] = (byte) value;
        }
        return getCRC(para);
    }


    /**
     * 计算CRC16校验码
     *
     * @param bytes 字节数组
     * @return {@link String} 校验码
     * @since 1.0
     */
    public static String getCRC(byte[] bytes) {
        //CRC寄存器全为1
        int CRC = 0x0000ffff;
        //多项式校验值
//        CRC16默认低前后高
        int POLYNOMIAL = 0x0000a001;
        int i, j;
        for (i = 0; i < bytes.length; i++) {
            CRC ^= (bytes[i] & 0x000000ff);
            for (j = 0; j < 8; j++) {
                if ((CRC & 0x00000001) != 0) {
                    CRC >>= 1;
                    CRC ^= POLYNOMIAL;
                } else {
                    CRC >>= 1;
                }
            }
        }
        //结果转换为16进制
        String result = Integer.toHexString(CRC).toUpperCase();
        if (result.length() != 4) {
            StringBuffer sb = new StringBuffer("0000");
            result = sb.replace(4 - result.length(), 4, result).toString();
        }
        //交换高低位  modbus为高前低后
        return result.substring(2, 4) + result.substring(0, 2);
    }


    public static void main(String[] args) {
        System.out.println(getCRC("01 03 10 A1 F1 00 77 0B 18 5C 5E 01 1A 00 00 00 00 00 01"));
    }
}
### 回答1: CRC(Cyclic Redundancy Check)循环冗余校验是一种简单且高效的错误检测算法,广泛应用于数据传输中的错误检测和校正。 CRC8和CRC16是两种常用的CRC校验算法。其中,CRC8算法用于校验8位字节或数据流的校验和,CRC16算法则用于校验16位字节或数据流的校验和。 CRC16算法CRC8算法更复杂,但CRC16的校验结果更可靠。CRC16算法在计算校验和时,使用一个16位的生成多项式,并且每次处理一个16位数据流。而CRC8算法则使用一个8位的生成多项式,每次处理一个8位数据流。 具体来说,CRC8算法首先需要一个初始值,然后依次读入每个数据字节,进行位运算和异或操作,最终计算出一个8位的校验和。CRC16算法与其类似,只不过初始化值更大,多项式也更长,计算结果也是一个16位的校验和。 虽然CRC算法的实现较为复杂,但它被广泛运用于通信和数据传输领域,因为它可以快速检测并校正数据传输过程中出现的错误。 ### 回答2: CRC算法是一种很常见的校验算法,其中最常见的两种是crc8和crc16算法CRC8算法是将传输的数据进行异或运算,并采取模2除法,将结果存储于一个8位寄存器中,最后输出8位的校验码。这种算法主要用于串行通信协议、存储设备和传感器网络。 CRC16算法是将输入数据除以一个特定的多项式,余数即为校验码。这种算法具有很高的误检率,可以在大多数场合用于通信的数据完整性检验。常见的应用包括:以太网、通讯协议、Modbus、X.25、SD卡等。与CRC8相比,CRC16可以检测更多的错误位。 在实际应用中,CRC算法通常需要选取合适的生成多项式,来保证算法的稳定性和准确性。同时,为了避免非正常条件下的错误数据干扰,可以在校验码中加入一定的冗余信息。 ### 回答3: CRC(Cyclic Redundancy Check)是一种校验算法,主要用于检测数据传输的出错情况。在计算机通信、存储等领域广泛应用。CRC算法可以检测出数据传输过程中的单比特差错和位移差错,但是它并不能检测出所有的差错。CRC算法是一种哈希函数的变种,通常采用多项式计算方法。 CRC算法的计算可以分为两个部分:生成表格和计算crc值。生成表格是为了在计算crc过程中快速地查找异或表格的值,而不必每次都进行一次多项式除法。生成表格的方法是将0~255的所有值带入多项式中进行计算,得到一个256位的表格。计算crc值时,将数据流和发送方预设的一个初始值一起丢进异或表格,对每一位进行异或,最后得到的结果就是crc值。 CRC算法有各种各样的规范,如CRC8、CRC16等,不同的规范所采用的多项式也不尽相同。其中,CRC8是一种8位循环冗余校验码,常用于通信协议中,如Modbus、I2C。CRC16是一种16位循环冗余校验码,常用于串口协议和Modbus RTU。 基于不同的多项式,CRC8和CRC16的校验强度也不同。通常来说,CRC16的校验强度要比CRC8高。但是由于CRC16需要计算的多项式位数较多,所以在计算速度上要比CRC8慢。因此,在实际应用中,需要根据具体的场景选择不同的CRC算法。 总之,CRC8和CRC16算法是数据通信中经常使用的一种校验算法。它能够快速检测出数据传输过程中的错误,保证数据的完整性和可靠性。在实际应用中,需要根据不同的场景和要求选择不同的CRC算法,以实现更好的校验效果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值