CRC16校验算法的Java实现

CRC16校验算法简介

CRC即循环冗余校验码(Cyclic Redundancy Check[1] ):是数据通信领域中最常用的一种查错校验码,其特征是信息字段和校验字段的长度可以任意选定。循环冗余检查(CRC)是一种数据传输检错功能,对数据进行多项式计算,并将得到的结果附在帧的后面,接收设备也执行类似的算法,以保证数据传输的正确性和完整性。

Crc16Util说明

CRC16有多种实现算法,这里是基于Modbus CRC16的校验算法的java实现。如果是基于其他的协议的算法,只需更改getCrc16()中的实现。
getCrc16();是核心算法,获取校验码byte数组。
intToBytes();将算出的int类型转成byte数组,低位在前,高位在后。改变高地位顺序,只需改变方法中数组的顺序。
getData();获取源数据和验证码的组合byte数组,可以传入byte数组,也可以传入十六进制字符数组。
byteTo16String();将byte或byte数组转换成十六进制字符,这里主要用于测试观察校验算法的结果。

工具类源码
/**
 * 基于Modbus CRC16的校验算法工具类
 */
public class Crc16Util {

    /**
     * 获取源数据和验证码的组合byte数组
     * @param strings 可变长度的十六进制字符串
     * @return
     */
    public static byte[] getData(String...strings) {
        byte[] data = new byte[]{};
        for (int i = 0; i<strings.length;i++) {
            int x = Integer.parseInt(strings[i], 16);
            byte n = (byte)x;
            byte[] buffer = new byte[data.length+1];
            byte[] aa = {n};
            System.arraycopy( data,0,buffer,0,data.length);
            System.arraycopy( aa,0,buffer,data.length,aa.length);
            data = buffer;
        }
        return getData(data);
    }
    /**
     * 获取源数据和验证码的组合byte数组
     * @param aa 字节数组
     * @return
     */
    private static byte[] getData(byte[] aa) {
        byte[] bb = getCrc16(aa);
        byte[] cc = new byte[aa.length+bb.length];
        System.arraycopy(aa,0,cc,0,aa.length);
        System.arraycopy(bb,0,cc,aa.length,bb.length);
        return cc;
    }
    /**
     * 获取验证码byte数组,基于Modbus CRC16的校验算法
     */
    private static byte[] getCrc16(byte[] arr_buff) {
        int len = arr_buff.length;

        // 预置 1 个 16 位的寄存器为十六进制FFFF, 称此寄存器为 CRC寄存器。
        int crc = 0xFFFF;
        int i, j;
        for (i = 0; i < len; i++) {
            // 把第一个 8 位二进制数据 与 16 位的 CRC寄存器的低 8 位相异或, 把结果放于 CRC寄存器
            crc = ((crc & 0xFF00) | (crc & 0x00FF) ^ (arr_buff[i] & 0xFF));
            for (j = 0; j < 8; j++) {
                // 把 CRC 寄存器的内容右移一位( 朝低位)用 0 填补最高位, 并检查右移后的移出位
                if ((crc & 0x0001) > 0) {
                    // 如果移出位为 1, CRC寄存器与多项式A001进行异或
                    crc = crc >> 1;
                    crc = crc ^ 0xA001;
                } else
                    // 如果移出位为 0,再次右移一位
                    crc = crc >> 1;
            }
        }
        return intToBytes(crc);
    }
    /**
     * 将int转换成byte数组,低位在前,高位在后
     * 改变高低位顺序只需调换数组序号
     */
    private static byte[] intToBytes(int value)  {
        byte[] src = new byte[2];
        src[1] =  (byte) ((value>>8) & 0xFF);
        src[0] =  (byte) (value & 0xFF);
        return src;
    }
    /**
     * 将字节数组转换成十六进制字符串
     */
    public static String byteTo16String(byte[] data) {
        StringBuffer buffer = new StringBuffer();
        for (byte b : data) {
            buffer.append(byteTo16String(b));
        }
        return buffer.toString();
    }
    /**
     * 将字节转换成十六进制字符串
     * int转byte对照表
     * [128,255],0,[1,128)
     * [-128,-1],0,[1,128)
     */
    public static String byteTo16String(byte b) {
        StringBuffer buffer = new StringBuffer();
        int aa = (int)b;
        if (aa<0) {
            buffer.append(Integer.toString(aa+256, 16)+" ");
        }else if (aa==0) {
            buffer.append("00 ");
        }else if (aa>0 && aa<=15) {
            buffer.append("0"+Integer.toString(aa, 16)+" ");
        }else if (aa>15) {
            buffer.append(Integer.toString(aa, 16)+" ");
        }
        return buffer.toString();
    }
}
测试代码
public static void main(String[] args) {

    byte[] dd = Crc16Util.getData("FE", "05","66");
    String str = Crc16Util.byteTo16String(dd).toUpperCase();
    System.out.println(str);
}

打印结果:FE 05 66 93 4A

  • 7
    点赞
  • 44
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
### 回答1: Java CRC16校验算法是一种常用的校验算法,用于检测数据传输或存储中的错误。CRC16算法生成一个16位的循环冗余校验码,用于验证数据完整性。 CRC (Cyclic Redundancy Check) 是一种循环冗余校验技术,通过多项式除法运算来生成校验码。CRC16是其中的一种算法,它使用16位的校验码。 Java实现CRC16校验算法可以使用位操作和位移运算,步骤如下: 1. 创建一个初始值为0的16位整数变量crc。 2. 遍历待校验的数据,对每个字节进行处理。 3. 将当前字节和crc的低8位进行异或操作,并将结果存回crc。 4. 循环8次,每次右移1位,并检查最低位是否为1。 - 如果最低位为1,则将crc与0xA001进行异或操作。 - 如果最低位为0,则不进行异或操作。 5. 重复第2步至第4步,直到处理完所有字节。 6. 返回最终得到的crc结果,即为校验码。 Java实现CRC16校验算法的代码示例: ```java public static int calculateCRC16(byte[] data) { int crc = 0x0000; for (byte b : data) { crc ^= (int) b & 0xFF; for (int i = 0; i < 8; i++) { if ((crc & 0x0001) != 0) { crc = (crc >> 1) ^ 0xA001; } else { crc >>= 1; } } } return crc; } ``` 以上就是Java CRC16校验算法的简要介绍和实现示例,通过这个算法可以有效地检测数据传输或存储中的错误。 ### 回答2: CRC16是一种校验算法,用于对数据进行校验和验证。它通过将数据转化成二进制位,并进行多项式取模运算来计算校验和。 CRC16算法的运算步骤如下: 1. 初始化校验值为0xFFFF。 2. 将待校验的数据按位转换为二进制形式。 3. 从高位开始,依次进行以下操作: - 将校验值与当前位异或。 - 如果结果的最高位为1,则将结果与一个预设的固定数值0x1021异或,否则什么都不做。 - 将结果左移一位。 4. 重复第3步,直到所有位都处理完毕。 5. 最后,将校验值取反,得到最终的校验和。 例如,假设待校验的数据为0xA1B2C3D4E5F6,将其转换为二进制形式为101000011011001011000011110100111011010111100110。 根据CRC16算法进行计算,最终得到的校验和为0xEA4D。 CRC16校验算法在网络通信、数据传输等领域中被广泛应用。其优点是计算速度快、实现简单,并且在数据传输过程中可以检测出大部分的传输错误。但需要注意的是,CRC16算法并不能保证100%的可靠性,只能对错误进行检测和部分纠正。 在Java中,可以使用现有的CRC16库来进行CRC16校验的计算。也可以自己实现算法,通过位运算和异或操作来完成校验和的计算。 ### 回答3: CRC16是一种校验算法,用于检测数据传输过程中的错误。它通过对数据进行计算,生成一个16位的校验码。CRC16算法是广泛应用于通信领域的一种校验方式。 CRC16算法的原理比较简单,主要通过多项式除法来实现。首先,将需要进行校验的数据按照特定的规则来生成一个二进制串,然后通过不断地进行异或和右移操作,最终得到一个16位的校验码。 具体实现CRC16算法的过程如下: 1. 初始化一个16位的寄存器为FFFFH,即所有位都为1。 2. 将待校验的数据按位进行异或操作,并将结果与寄存器进行异或。 3. 对寄存器进行右移一位,丢弃最低位,如果最低位为1,则与一个特定的多项式0xA001进行异或。 4. 重复2和3步骤,直到所有数据位都经过了异或操作。 5. 最后,得到的16位寄存器的内容就是CRC16校验码。 CRC16算法具有较高的校验能力和良好的检错性能,因此在通信中被广泛应用。它可以有效地检测数据传输中的错误,保证数据的完整性和正确性。 总的来说,CRC16是一种简单且高效的校验算法,适用于各种数据校验的场景。在Java中,可以通过使用位运算和逻辑运算等操作来实现CRC16算法的计算。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值