JAVA CRC16校验算法

JAVA Modbus CRC16 校验码生成算法

参考文档

校验码: C R C 1 6 ( 1 ) \color{#FF0000}{CRC16^{(1)}} CRC16(1) 占用两个字节,包含了一个 16 位的二进制值。CRC 值由传输设备计算出来,
然后附加到数据帧上,接收设备在接收数据时重新计算 CRC 值,然后与接收到的 CRC 域中的值
进行比较,如果这两个值不相等,就发生了错误。
生成一个 CRC16 的流程为:
(1) 预置一个 16 位寄存器为 0 F F F F H ( 2 ) \color{#FF0000}{0FFFFH}^{(2)} 0FFFFH(2)(全 1),称之为 CRC 寄存器。
(2) 把数据帧中的第一个字节的 8 位与 CRC 寄存器中的低字节进行异或运算,结果存回
CRC 寄存器。
(3) 将 CRC 寄存器向右移一位,最高位填以 0,最低位移出并检测。
(4) 如果最低位为 0:重复第三步(下一次移位);如果最低位为 1:将 CRC 寄存器与一个
预设的固定值( 0 A 001 H ( 3 ) \color{#FF0000}{0A001H}^{(3)} 0A001H(3))进行异或运算。
(5) 重复第三步和第四步直到 8 次移位。这样处理完了一个完整的八位。
(6) 重复第 2 步到第 5 步来处理下一个八位,直到所有的字节处理结束。
(7) 最终 CRC 寄存器的值就是 CRC16 的值。

请注意上述文档中说明的
(1) CRC宽度
(2) 初始值
(3) 多项式

请根据具体项目要求修改修改算法

算法代码

/**
 * Create by 郭文梁 2019/6/25 0025 10:32
 * CRC16
 * CRC16工具类
 *
 * @author 郭文梁
 * @data 2019/6/25 0025
 */
public class CRC {
    /**
     * 一个字节包含位的数量 8
     */
    private static final int BITS_OF_BYTE = 8;
    /**
     * 多项式
     */
    private static final int POLYNOMIAL = 0xA001;
    /**
     * 初始值
     */
    private static final int INITIAL_VALUE = 0xFFFF;

    /**
     * CRC16 编码
     *
     * @param bytes 编码内容
     * @return 编码结果
     */
    public static int crc16(int[] bytes) {
        int res = INITIAL_VALUE;
        for (int data : bytes) {
            res = res ^ data;
            for (int i = 0; i < BITS_OF_BYTE; i++) {
                res = (res & 0x0001) == 1 ? (res >> 1) ^ POLYNOMIAL : res >> 1;
            }
        }
        return revert(res);
    }

    /**
     * 翻转16位的高八位和低八位字节
     *
     * @param src 翻转数字
     * @return 翻转结果
     */
    private static int revert(int src) {
        int lowByte = (src & 0xFF00) >> 8;
        int highByte = (src & 0x00FF) << 8;
        return lowByte | highByte;
    }
}

补充说明

由于文档中提到如下内容

数据帧的结构,即报文格式:

设备地址功能代码数据段CRC16校验码
1个byte1个byteN个byte2个byte( 低 字 节 在 前 \color{#FF0000}{低字节在前} )

故代码中使用了 revert 函数翻转了16位CRC校验值的高低位

测试代码

参考测试结果
参考测试结果
测试代码

package club.xyes.zkh.retail.commons.utils;

import lombok.extern.slf4j.Slf4j;
import org.junit.Test;

@Slf4j
public class CRC16Test {
    @Test
    public void test() {
        final int[] data = {
                0x01,
                0x10,
                0x00,
                0x0C,
                0x00,
                0x02,
                0x04,
                0x00,
                0x00,
                0x00,
                0x00
        };
        final int res = CRC.crc16(data);
        final String hex = Integer.toHexString(res);
        log.debug("Res hex {}", hex);
    }
}

输出

13:41:48.397 [main] DEBUG club.xyes.zkh.retail.commons.utils.CRC16Test - Res hex f3fa
  • 3
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答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算法的计算。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值