ModbusCRC16校验 示例代码

作者: Herman Ye @Galbot @Auromix
测试环境: Ubuntu20.04
更新日期: 2023/08/30
1 @Auromix 是一个机器人爱好者开源组织。
2 本文在更新日期经过测试,确认有效。

笔者出于学习交流目的,
给出以下ModbusCRC16校验常用的四种函数以及完整示例代码:

1.计算CRC

注意: 此处在末尾进行了高低位交换,可根据需求删减代码交换高低位顺序。

/*** @brief Calculate the Modbus CRC-16 checksum for a given data vector.** This function calculates the CRC-16 checksum using the Modbus protocol* polynomial for the provided data vector.** @param data The vector of bytes for which to calculate the CRC-16 checksum.* @return The calculated CRC-16 checksum value as an unsigned short.*/
unsigned short calculateModbusCRC16(const vector<uint8_t> &data)
{int length = data.size();unsigned short CRC = 0xFFFF; // Initial valuefor (int i = 0; i < length; i++){CRC = CRC ^ data[i]; // XOR byte into the least significant byte of CRCfor (int j = 0; j < 8; j++){if (CRC & 1){CRC >>= 1;CRC ^= 0xA001;}else{CRC >>= 1;}}}// Swap the bytes of the CRC to match Modbus conventionunsigned short swappedCRC = ((CRC >> 8) & 0xFF) | ((CRC & 0xFF) << 8);return swappedCRC;
}

2.添加CRC校验位

注意: 此处进行了高低位交换,可根据需求删减代码交换高低位顺序。

/*** @brief Add Modbus CRC-16 to a data vector.* * This function calculates the Modbus CRC-16 checksum for the provided data vector* and appends the high and low bytes of the CRC to the end of the data vector.* * @param data The data vector to which the CRC will be added.*/
void addModbusCRC16(vector<uint8_t> &data)
{// Calculate the CRC-16 checksumunsigned short crc = calculateModbusCRC16(data);// Append the high byte of CRC to the data vectordata.push_back((crc >> 8) & 0xFF);// Append the low byte of CRC to the data vectordata.push_back(crc & 0xFF);
}

3.删除CRC校验位

/*** @brief Remove Modbus CRC-16 from a vector of data.* * This function takes a vector of data with Modbus CRC-16 at the end and removes* the CRC-16 bytes from the end of the vector, effectively stripping the CRC-16* checksum from the data.* * @param dataWithCRC A reference to the vector containing the data with CRC-16.* * @note This function does not perform any CRC-16 validation or calculation. It* assumes that the last two bytes of the vector represent the CRC-16 checksum* and removes them regardless of their validity.* * @warning It is the responsibility of the caller to ensure that the input vector* has a length of at least 2, as this function does not perform length checking.* If the length is less than 2, an error message is printed to the standard error* stream, and no modifications are made to the input vector.*/
void removeModbusCRC16(vector<uint8_t> &dataWithCRC) {int length = dataWithCRC.size();// Error checkif (length < 2) {cerr << "Invalid data length" << endl;return;}// Delete CRC at the enddataWithCRC.resize(length - 2);
}

4.比较CRC校验位

/*** @brief Compare Modbus CRC-16* * This function compares the CRC-16 checksum in the given data with the calculated CRC-16 checksum* of the data without the CRC bytes. If they match, it indicates that the data is intact and has not* been corrupted during transmission.* * @param dataWithCRC A vector containing the data along with the CRC-16 checksum bytes.* @return True if the calculated CRC-16 matches the original CRC-16, indicating data integrity.*         False if the data length is invalid or if the CRCs do not match.*/
bool compareModbusCRC16(const vector<uint8_t> &dataWithCRC) {int length = dataWithCRC.size();// Error checkif (length < 2) {cerr << "Invalid data length" << endl;return false;}// Get data without CRCvector<uint8_t> dataWithoutCRC(dataWithCRC.begin(), dataWithCRC.end() - 2);// Calculate CRC-16 checksumunsigned short calculatedCRC = calculateModbusCRC16(dataWithoutCRC);// Get original CRC-16 checksum from the last two bytes of the dataunsigned short originalCRC = (dataWithCRC[length - 2] << 8) | dataWithCRC[length - 1];return originalCRC == calculatedCRC;
}

5.完整示例代码

#include <iostream>
#include <vector>
#include <iomanip>
using namespace std;/*** @brief Calculate the Modbus CRC-16 checksum for a given data vector.** This function calculates the CRC-16 checksum using the Modbus protocol* polynomial for the provided data vector.** @param data The vector of bytes for which to calculate the CRC-16 checksum.* @return The calculated CRC-16 checksum value as an unsigned short.*/
unsigned short calculateModbusCRC16(const vector<uint8_t> &data)
{int length = data.size();unsigned short CRC = 0xFFFF; // Initial valuefor (int i = 0; i < length; i++){CRC = CRC ^ data[i]; // XOR byte into the least significant byte of CRCfor (int j = 0; j < 8; j++){if (CRC & 1){CRC >>= 1;CRC ^= 0xA001;}else{CRC >>= 1;}}}// Swap the bytes of the CRC to match Modbus conventionunsigned short swappedCRC = ((CRC >> 8) & 0xFF) | ((CRC & 0xFF) << 8);return swappedCRC;
}/*** @brief Add Modbus CRC-16 to a data vector.* * This function calculates the Modbus CRC-16 checksum for the provided data vector* and appends the high and low bytes of the CRC to the end of the data vector.* * @param data The data vector to which the CRC will be added.*/
void addModbusCRC16(vector<uint8_t> &data)
{// Calculate the CRC-16 checksumunsigned short crc = calculateModbusCRC16(data);// Append the high byte of CRC to the data vectordata.push_back((crc >> 8) & 0xFF);// Append the low byte of CRC to the data vectordata.push_back(crc & 0xFF);
}/*** @brief Remove Modbus CRC-16 from a vector of data.* * This function takes a vector of data with Modbus CRC-16 at the end and removes* the CRC-16 bytes from the end of the vector, effectively stripping the CRC-16* checksum from the data.* * @param dataWithCRC A reference to the vector containing the data with CRC-16.* * @note This function does not perform any CRC-16 validation or calculation. It* assumes that the last two bytes of the vector represent the CRC-16 checksum* and removes them regardless of their validity.* * @warning It is the responsibility of the caller to ensure that the input vector* has a length of at least 2, as this function does not perform length checking.* If the length is less than 2, an error message is printed to the standard error* stream, and no modifications are made to the input vector.*/
void removeModbusCRC16(vector<uint8_t> &dataWithCRC) {int length = dataWithCRC.size();// Error checkif (length < 2) {cerr << "Invalid data length" << endl;return;}// Delete CRC at the enddataWithCRC.resize(length - 2);
}/*** @brief Compare Modbus CRC-16* * This function compares the CRC-16 checksum in the given data with the calculated CRC-16 checksum* of the data without the CRC bytes. If they match, it indicates that the data is intact and has not* been corrupted during transmission.* * @param dataWithCRC A vector containing the data along with the CRC-16 checksum bytes.* @return True if the calculated CRC-16 matches the original CRC-16, indicating data integrity.*         False if the data length is invalid or if the CRCs do not match.*/
bool compareModbusCRC16(const vector<uint8_t> &dataWithCRC) {int length = dataWithCRC.size();// Error checkif (length < 2) {cerr << "Invalid data length" << endl;return false;}// Get data without CRCvector<uint8_t> dataWithoutCRC(dataWithCRC.begin(), dataWithCRC.end() - 2);// Calculate CRC-16 checksumunsigned short calculatedCRC = calculateModbusCRC16(dataWithoutCRC);// Get original CRC-16 checksum from the last two bytes of the dataunsigned short originalCRC = (dataWithCRC[length - 2] << 8) | dataWithCRC[length - 1];return originalCRC == calculatedCRC;
}int main() {// Example data 1vector<uint8_t> deviceData1 = {0x01, 0x10, 0x00, 0x02, 0x00, 0x06, 0x0C, 0x41, 0x20,0x00, 0x00, 0x42, 0xC8, 0x00, 0x00, 0x42, 0x48, 0x00, 0x00,0x84, 0xC1}; // Example CRC: 0x84, 0xC1// Print original datacout << "Original data 1: ";for (uint8_t byte : deviceData1) {cout << hex << uppercase << setw(2) << setfill('0') << (int)byte << " ";}cout << endl;bool comparedResult=compareModbusCRC16(deviceData1);if (comparedResult)cout<<"Compared result: "<<"TRUE"<<endl;elsecout<<"Compared result: "<<"FALSE"<<endl;// Example data 2cout<<endl;vector<uint8_t> deviceData2 = {0x01, 0x06, 0x00, 0x00, 0x01, 0x02, 0x02};// Example CRC: 0xDA, 0xC7cout << "Original data 2: ";for (uint8_t byte : deviceData2) {cout << hex << uppercase << setw(2) << setfill('0') << (int)byte << " ";}cout << endl;// Add CRC and print modified dataaddModbusCRC16(deviceData2);cout << "Add CRC to original data 2: ";for (uint8_t byte : deviceData2) {cout << hex << uppercase << setw(2) << setfill('0') << (int)byte << " ";}cout << endl;// Remove CRC from dataremoveModbusCRC16(deviceData2);cout << "Remove CRC from modified data 2: ";for (uint8_t byte : deviceData2) {cout << hex << uppercase << setw(2) << setfill('0') << (int)byte << " ";}cout << endl;	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/67745.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

51单片机DHT11温湿度控制系统仿真设计( proteus仿真+程序+原理图+报告+讲解视频)

51单片机DHT11温湿度控制系统仿真设计 1.主要功能&#xff1a;2.仿真3. 程序代码4. 原理图元器件清单5. 设计报告6. 设计资料内容清单&下载链接 51单片机DHT11温湿度控制系统仿真设计( proteus仿真程序原理图报告讲解视频&#xff09; 仿真图proteus8.9及以上 程序编译器&…

Css 将某div设置为透明,但其子元素不透明

可以使用css中的属性opacity来设置元素的透明度&#xff0c;但它会影响到元素的所有子元素。如果想让父元素透明&#xff0c;但子元素不透明&#xff0c;可以使用另外一种方法&#xff1a; 首先&#xff0c;将父元素的背景颜色设置为rgba格式&#xff0c;其中a表示透明度。例如…

IDEA快捷键

crtlshiftf8 查看所有IDEA debug代码的位置

云备份项目

云备份项目 1. 云备份认识 自动将本地计算机上指定文件夹中需要备份的文件上传备份到服务器中。并且能够随时通过浏览器进行查看并且下载&#xff0c;其中下载过程支持断点续传功能&#xff0c;而服务器也会对上传文件进行热点管理&#xff0c;将非热点文件进行压缩存储&…

爬虫项目(二):中国大学排名

《Python网络爬虫入门到实战》京东购买地址&#xff0c;这里讲解了大量的基础知识和实战&#xff0c;由本人编著&#xff1a;https://item.jd.com/14049708.html配套代码仓库地址&#xff1a;https://github.com/sfvsfv/Crawer文章目录 分析第一步&#xff1a;获取源码分析第一…

【嵌入式软件开发 】学习笔记

本文主要记录 【嵌入式软件开发】 学习笔记&#xff0c;参考相关大佬的文章 1.RTOS 内功修炼笔记 RTOS内功修炼记&#xff08;一&#xff09;—— 任务到底应该怎么写&#xff1f; RTOS内功修炼记&#xff08;二&#xff09;—— 优先级抢占式调度到底是怎么回事&#xff1f;…

kafka复习:(26)通过RecordHeaders和RecordHeader来实现TTL功能

一、定义生产者&#xff0c;在消息中加入RecordHeaders package com.cisdi.dsp.modules.metaAnalysis.rest.kafka2023;import org.apache.kafka.clients.producer.KafkaProducer; import org.apache.kafka.clients.producer.ProducerConfig; import org.apache.kafka.clients.…

【力扣】304. 二维区域和检索 - 矩阵不可变 <二维前缀和>

目录 【力扣】304. 二维区域和检索 - 矩阵不可变二维前缀和理论初始化计算面积 题解 【力扣】304. 二维区域和检索 - 矩阵不可变 给定一个二维矩阵 matrix&#xff0c;以下类型的多个请求&#xff1a; 计算其子矩形范围内元素的总和&#xff0c;该子矩阵的 左上角 为 (row1, …

界面控件DevExtreme(v23.2)下半年发展路线图

在这篇文章中&#xff0c;我们将介绍DevExtreme在v23.2中发布的一些主要特性&#xff0c;这些特性既适用于DevExtreme JavaScript (Angular、React、Vue、jQuery)&#xff0c;也适用于基于DevExtreme的ASP. NET MVC/Core控件。 DevExtreme包含全面的高性能和响应式UI小部件集合…

netty(三):NIO——多线程优化

NIO多线程优化 使用Boss线程来处理accepct事件使用Worker线程来处理读写事件&#xff0c;可以创建多个worker线程 package com.review;import lombok.extern.slf4j.Slf4j;import java.io.IOException; import java.net.InetSocketAddress; import java.nio.channels.*; impor…

elementui的el-dialog组件与el-tabs同时用导致浏览器卡死的原因解决

文章目录 在el-dialog弹出框中&#xff0c;如果使用el-tabs&#xff0c;点击弹框的关闭按钮时&#xff0c;会导致弹出框无法关闭&#xff0c;且浏览器卡死。 环境是&#xff1a; vue :2.6.10 elementui:2.15.3解决方式&#xff1a; 在el-talbs的面板中不放任何内容&#xff…

需求分析入门

认识管理软件 什么是管理软件 管理软件就是用来辅助企业进行管理的软件&#xff0c;既包括对企业“人、财、物”相关的资产信息的管理&#xff0c;也包括对企业“供、产、销”相关的业务活动信息的管理。管理软件的重点在于管理信息的收集、流转&#xff0c;资源的共享、集成…

UML基础

统一建模语言&#xff08;UML是 Unified Modeling Language的缩写, 是用来对软件系统进行可视化建模的一种语言。UML为面向对象开发系统的产品 进行说明、可视化、和编制文档的一种标准语言。 共有9种图 UML中的图其实不止九种 (相同的图还可能会有不同的名称), 这里的九种图是…

Ubuntu 20.04 网卡命名规则

Ubuntu 系统中网卡的命名规则是&#xff1a;Consistent Network Device Naming&#xff08;一致网络设备命名&#xff09;规范。这个规范的原理是根据固件、拓扑和位置信息分配固定名称。其中&#xff0c;设备类型 en 代表 Ethernet (以太网)&#xff0c;wl 代表 WLAN&#xff…

Linux(基础IO)

Linux&#xff08;基础IO&#xff09; 前言C语言文件IO什么叫当前路径stdin/stdout/stderr 系统文件IOopenclosewriteread 文件描述符文件描述符的分配规则 重定向输出重定向原理追加重定向原理输入重定向原理dup2添加重定向功能到minishell 缓冲区模拟实现一个缓冲区 理解文件…

云原生周刊:Linkerd 发布 v2.14 | 2023.9.4

开源项目推荐 Layerform Layerform 是一个 Terraform 包装器&#xff0c;可帮助工程师使用纯 Terraform 文件构建可重用的基础设施。 为了实现重用&#xff0c;Layerform 引入了层的概念。每层都包含一些基础设施&#xff0c;并且可以堆叠在另一层之上。 除了更易于使用之外…

RK3568-android11-适配ov13850摄像头

硬件连接 主要分为两部分: mipi接口:传输摄像头数据 i2c接口:配置摄像头和对焦马达芯片寄存器相关驱动 |-- arch/arm64/boot/dts/rockchip DTS配置文件 |-- drivers/phy/rockchip/|-- phy-rockchip-mipi-rx.c mipi dphy 驱动 |-- drivers/media||-- platform/rockchip/isp1…

uniapp 集成蓝牙打印功能(个人测试佳博打印机)

uniapp 集成蓝牙打印功能&#xff08;个人测试京博打印机&#xff09; uniapp 集成蓝牙打印功能集成佳博内置的接口 uniapp 集成蓝牙打印功能 大家好今天分析的是uniapp 集成蓝牙打印功能&#xff0c;个人开发是app,应该是支持H5(没试过) 集成佳博内置的接口 下载dome地址&…

2023高教社杯数学建模A题B题C题D题E题思路模型 国赛建模思路分享

文章目录 0 赛题思路1 竞赛信息2 竞赛时间3 建模常见问题类型3.1 分类问题3.2 优化问题3.3 预测问题3.4 评价问题 4 建模资料 0 赛题思路 &#xff08;赛题出来以后第一时间在CSDN分享&#xff09; https://blog.csdn.net/dc_sinor?typeblog 1 竞赛信息 全国大学生数学建模…

怎么把pdf压缩的小一点?

怎么把pdf压缩的小一点&#xff1f;在我们日常的学习和工作中&#xff0c;PDF文件是一个非常常见和有用的文件格式&#xff0c;并且受到很多小伙伴的喜欢。有时候&#xff0c;一些PDF文件可能会很大&#xff0c;造成pdf文件较大的原因其实很明确&#xff0c;主要是因为pdf文件中…