资讯专栏INFORMATION COLUMN

Java CRC16 MODBUS校验算法实现

mindwind / 1953人阅读

CRC16 MODBUS校验算法,分两种情况,一是参数是字符串,二是参数是字节数组。
参考文章1
参考文章2
直接上代码,第一种参数是字符串

public static void main(String[] args) throws IOException {
    String frame = "01 03 01 48 00 0c";
    System.out.println(getCRC(frame));
}

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;
    // 多项式校验值
    int POLYNOMIAL = 0x0000a001;
    int i, j;
    for (i = 0; i < bytes.length; i++) {
        CRC ^= ((int) 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();
    }
    //高位在前地位在后
    //return result.substring(2, 4) + " " + result.substring(0, 2);
    // 交换高低位,低位在前高位在后
    return result.substring(2, 4) + " " + result.substring(0, 2);
}

执行后结果:

第二种参数是byte数组:

public static void main(String[] args) throws IOException {
    byte[] bytes = new byte[] { 0x01, 0x03, 0x01, 0x48, 0x00, 0x0c };
    System.out.println(getCRC2(bytes));
}

public static String getCRC2(byte[] bytes) {
    int CRC = 0x0000ffff;
    int POLYNOMIAL = 0x0000a001;

    int i, j;
    for (i = 0; i < bytes.length; i++) {
        CRC ^= (int) bytes[i];
        for (j = 0; j < 8; j++) {
            if ((CRC & 0x00000001) == 1) {
                CRC >>= 1;
                CRC ^= POLYNOMIAL;
            } else {
                CRC >>= 1;
            }
        }
    }
    // 交换高低位,低位在前高位在后
    CRC = ((CRC & 0x0000FF00) >> 8) | ((CRC & 0x000000FF) << 8);
    String result = Integer.toHexString(CRC);
    return result.substring(0, 2) + " " + result.substring(2, 4);
}

运行结果:

小菜一枚,不正确之处还请各位指教!

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/74812.html

相关文章

  • 【RISC-V MCU 创新应用】一种网络型单相智能电表

    摘要:负责对电能检测模块下发收集指令,以及对电能检测模块响应的数据进行处理,一方面对接收的单相数据进行转换后驱动屏幕对相关数据进行显示,另一方面对单相数据进行重新打包并通过串口对以太网模块进行数据传输。 ...

    rockswang 评论0 收藏0
  • java.util.zip.CRC32和java.util.zip.Adler32的性能对比分析

    摘要:当主机正在从某个输入流源源不断地读取文件时,校验码的计算是实时的,每次校验操作都只对文件的一部分进行处理。在计算校验码时,请至少使用的分区大小,否则调用会造成不可忽略的时间消耗。 原文:java.util.zip.CRC32 and java.util.zip.Adler32 performance作者:Mikhail Vorontsov 校验码是把任意长度的字节内容输入通过特定算法变...

    ISherry 评论0 收藏0
  • Java–串口之间的通信及扫描枪的读取

    摘要:串口通信以自己刚做的功能做为核心记录这篇博客是基于通信协议之间的通信是十六进制进行通信的可选波特率,数据位停止位采用校验功能和代码的实现界面是用窗体表现的界面的实现就不过多的贴代码和说怎么去实现了,有个很好用的工具,可以直接拖控件,跟做很像 串口通信以自己刚做的功能做为核心记录这篇博客①. 是基于modbus通信协议之间的通信②. 是十六进制进行通信的③. 可选波特率,数据位 停止位④...

    EsgynChina 评论0 收藏0
  • 通过PHP实现一致性哈希算法

    摘要:通过虚拟节点优化一致性算法为了提高一致性算法的平衡性,我们首先能够想到的是,增加节点数,但是机器毕竟是需要经费啊,不是说增就能随意增,那就增加虚拟节点,这样就没毛病了。 一、案例分析(1)问题概述 假设我们的图片数据均匀的分配在三台服务(分别标注为服务器A,服务器B、服务器C)上面,现在我们要从里面取图片,服务端在拿到这个请求后,怎么会指定,这张图片是存在服务器A、服务器B,还是服务器...

    tulayang 评论0 收藏0

发表评论

0条评论

mindwind

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<