CRC16校驗 Java 實現(xiàn)

 * 校驗碼:CRC16占用兩個字節(jié),包含了一個 16 位的二進(jìn)制值。CRC 值由傳輸設(shè)備計算出來,
 * 然后附加到數(shù)據(jù)幀上,接收設(shè)備在接收數(shù)據(jù)時重新計算 CRC 值,然后與接收到的 CRC 域中的值
 * 進(jìn)行比較,如果這兩個值不相等,就發(fā)生了錯誤。
 * 生成一個 CRC16 的流程為:
 * (1) 預(yù)置一個 16 位寄存器為 0FFFFH(全為1),稱之為 CRC 寄存器。
 * (2) 把數(shù)據(jù)幀中的第一個字節(jié)的 8 位與 CRC 寄存器中的低字節(jié)進(jìn)行異或運(yùn)算,結(jié)果存回 CRC 寄存器。
 * (3) 將 CRC 寄存器向右移一位,最高位填以 0,最低位移出并檢測。
 * (4) 如果最低位為 0:重復(fù)第三步(下一次移位);如果最低位為 1,將 CRC 寄存器與一個預(yù)設(shè)的固定值 0A001H 進(jìn)行異或運(yùn)算。
 * (5) 重復(fù)第三步和第四步直到 8 次移位。這樣處理完了一個完整的八位。
 * (6) 重復(fù)第 2 步到第 5 步來處理下一個八位,直到所有的字節(jié)處理結(jié)束。
 * (7) 最終 CRC 寄存器的值就是 CRC16 的值。
 */
public class CRC {

    /**
     * 一個字節(jié)包含位的數(shù)量 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 編碼內(nèi)容
     * @return 編碼結(jié)果
     */
    public static String 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 convertToHexString(revert(res));
    }

    /**
     * 翻轉(zhuǎn)16位的高八位和低八位字節(jié)
     *
     * @param src 翻轉(zhuǎn)數(shù)字
     * @return 翻轉(zhuǎn)結(jié)果
     */
    private static int revert(int src) {
        int lowByte = (src & 0xFF00) >> 8;
        int highByte = (src & 0x00FF) << 8;
        return lowByte | highByte;
    }

    private static String convertToHexString(int src) {
        return Integer.toHexString(src);
    }

    public static void main(String[] args) {

        int[] data = new int[]{0x01, 0x04, 0x04, 0x01, 0x0e, 0x01, 0xde};
        System.out.println(CRC.crc16(data));


        int b = 0x0201;

        // 將16位的高8位轉(zhuǎn)換為低8位
        int lowByte = (b & 0xFF00) >> 8;
        System.out.println(lowByte);

        // 將16位的低8位轉(zhuǎn)換為高8位
        int highByte = (b & 0x00FF) << 8;
        System.out.println(highByte);

        // 按位或運(yùn)算,將兩個數(shù)相加
        int c = lowByte | highByte;
        System.out.println(c);

    }
}



最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

友情鏈接更多精彩內(nèi)容