Qt BCC异或校验、LRC纵向冗余校验、CRC循环冗余校验

BCC异或校验
将每个字节数据(一般是两个16进制的字符)进行异或后即得到校验码。适用于大多数要求不高的数据通讯,像很多单片机系统的串口通讯。

ASill异或校验

unsigned char GetBCC(unsigned char *buf, unsigned int  bufLen)
{
    unsigned char check = 0; //校验字节 初始化
    while(bufLen--) check ^= *buf++; //异或计数
    return check;
}

十六进制异或校验

QString checkXorH(QByteArray data)
{
    QStringList strData;
    QString rdata = data;
    int  checksum = 0;
    for(int i = 0; i < rdata.size()/2; i++){
        strData.append(rdata.mid(i*2,2));
    }
    for (int i = 0;i < rdata.size()/2;i++)
    {
        checksum = checksum ^ strData[i].toInt(nullptr,16);
    }
    strData.clear();
    return QString::number(checksum,16).toUpper();
}

十进制异或校验

QString checkXorD(QString data)
{
    QStringList strData;
    int  checksum = 0;
    data = data.toLatin1().toHex();
    for(int i = 0; i < data.size()/2; i++){
        strData.append(data.mid(i*2,2));
    }
    for (int i = 0;i < data.size()/2;i++)
    {
        checksum = checksum ^ strData[i].toInt(nullptr,16);
    }
    strData.clear();
    return QString::number(checksum,16).toUpper();
}

LRC纵向冗余校验
将每个字节数据求和结果与256求余,再按位取反加 1(256 - 余数)后即得到校验码。在工业领域Modbus协议Ascii模式采用该算法


unsigned char GetLRC(unsigned char *buf, unsigned int bufLen)
{
    unsigned char check = 0; //校验字节 初始化
    while(bufLen--) check += *buf++ ; //累加求余(只保留余数字节就是数据求和结果与256求余)
    return (unsigned char)((check ^ 0xFF) + 1);
}

CRC循环冗余校验

#define FACTOR (0x107 & 0xFF) //多项式因子(取低8bit)
unsigned char GetCRC(unsigned char *pdat, unsigned int len)
{
    unsigned char j;
    unsigned char crc = 0x00;
    while(len--)
    {
        crc ^= (*pdat++);//前一字节计算CRC后的结果异或上后一字节,再次计算CRC
        for (j=8; j>0; j--)
        {
            if (crc & 0x80)//高位为1,需要异或;否则,不需要
            {
                crc = (crc << 1) ^ FACTOR;
            }
            else
            {
                crc = (crc << 1);
            }
        }
    }

    return crc;
}

你可能感兴趣的:(c/c++,QT,单片机,嵌入式硬件)