1. 程式人生 > >JAVA CRC16校驗碼計算

JAVA CRC16校驗碼計算

package com.hcs.lqjc.controller.dongDaHengFeng.ModBus.utils;

/**
 * @author lwt
 * @date 2018-06-26
 *
 * 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;
        //多項式校驗值
        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);
    }


    public static void main(String[] args) {
        //01 03 20 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 8C 45
        //01 03 00 00 00 08 44 0C
        //01 03 10 00 8F 02 4E 00 91 02 44 00 92 02 5A 00 8B 02 47 40 D8
        System.out.println(getCRC("01 03 20 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF 7F FF"));
        System.out.println(getCRC("01 03 00 00 00 08"));
        System.out.println(getCRC("01 03 10 00 8F 02 4E 00 91 02 44 00 92 02 5A 00 8B 02 47"));
    }
}