在Modbus RTU中同时进行读取和写入操作

简介: 在Modbus RTU中同时进行读取和写入操作
#include "ModbusRTU.h"
#include <iostream>
#include <cstring> // For memcpy

ModbusRTU::ModbusRTU(boost::asio::io_service& io_service, const std::string& port_name)
    : serial(io_service, port_name) {
    serial.set_option(boost::asio::serial_port_base::baud_rate(9600));
    serial.set_option(boost::asio::serial_port_base::character_size(8));
    serial.set_option(boost::asio::serial_port_base::parity(boost::asio::serial_port_base::parity::none));
    serial.set_option(boost::asio::serial_port_base::stop_bits(boost::asio::serial_port_base::stop_bits::one));
}

template<typename T>
void ModbusRTU::writeData(uint8_t slave_id, uint8_t function_code, uint16_t start_address, T value) {
    uint8_t frame[8 + sizeof(T)];
    frame[0] = slave_id;
    frame[1] = function_code;
    frame[2] = (start_address >> 8) & 0xFF;
    frame[3] = start_address & 0xFF;

    std::memcpy(&frame[4], &value, sizeof(T));

    uint16_t crc = calculateCRC(frame, 4 + sizeof(T));
    frame[4 + sizeof(T)] = crc & 0xFF;
    frame[5 + sizeof(T)] = (crc >> 8) & 0xFF;

    boost::asio::write(serial, boost::asio::buffer(frame, 6 + sizeof(T)));
}

template<typename T>
T ModbusRTU::readData(uint8_t slave_id, uint8_t function_code, uint16_t start_address, uint16_t num_registers) {
    uint8_t frame[8];
    frame[0] = slave_id;
    frame[1] = function_code;
    frame[2] = (start_address >> 8) & 0xFF;
    frame[3] = start_address & 0xFF;
    frame[4] = (num_registers >> 8) & 0xFF;
    frame[5] = num_registers & 0xFF;
    uint16_t crc = calculateCRC(frame, 6);
    frame[6] = crc & 0xFF;
    frame[7] = (crc >> 8) & 0xFF;

    boost::asio::write(serial, boost::asio::buffer(frame, 8));

    uint8_t response[256];
    size_t length = serial.read_some(boost::asio::buffer(response, 256));
    if (length > 0 && response[1] == function_code) {
        T result = 0;
        for (size_t i = 0; i < sizeof(T); ++i) {
            reinterpret_cast<uint8_t*>(&result)[i] = response[3 + i];
        }
        return result;
    }
    else {
        std::cerr << "错误:无效的响应或没有收到响应。" << std::endl;
    }
    return T(); // 返回类型 T 的默认值
}

uint16_t ModbusRTU::calculateCRC(const uint8_t* data, size_t length) {
    uint16_t crc = 0xFFFF;
    for (size_t i = 0; i < length; ++i) {
        crc ^= data[i];
        for (int j = 0; j < 8; ++j) {
            if (crc & 1) crc = (crc >> 1) ^ 0xA001;
            else crc >>= 1;
        }
    }
    return crc;
}



void main() {
    try {
        boost::asio::io_service io_service;
        ModbusRTU modbus(io_service, "/dev/ttyS0"); // 根据你的实际串口设备调整

        uint8_t slave_id = 1;  // 从站地址
        uint16_t start_address = 0x0001; // 起始寄存器地址
        uint16_t value_to_write = 12345;  // 要写入的值

        // 写入数据
        modbus.writeData<uint16_t>(slave_id, 0x06, start_address, value_to_write); // 0x06 是写单寄存器的功能码

        std::cout << "数据已写入。" << std::endl;

        // 读取数据
        uint16_t read_value = modbus.readData<uint16_t>(slave_id, 0x03, start_address, 1); // 0x03 是读保持寄存器的功能码

        std::cout << "读取的数据: " << read_value << std::endl;

    }
    catch (const std::exception& e) {
        std::cerr << "异常: " << e.what() << std::endl;
        return 1;
    }

    return 0;
}
目录
相关文章
|
存储 开发框架 算法
【串口通信】使用C++和Qt设计和实现串口协议解析器(一)
【串口通信】使用C++和Qt设计和实现串口协议解析器
4117 0
|
XML 数据格式 Python
旋转标注工具roLabelImg使用教程
旋转标注工具roLabelImg使用教程
旋转标注工具roLabelImg使用教程
|
5月前
|
关系型数据库 MySQL 分布式数据库
阿里云PolarDB云数据库收费价格:MySQL、Postgre和PostgreSQL费用清单
阿里云PolarDB兼容MySQL、PostgreSQL及Oracle语法,支持集中式与分布式架构。标准版2核4G年费低至1116元,企业版最高性能达4核32GB。具备高可用、弹性扩展、HTAP实时分析能力,广泛应用于金融、互联网、政务等领域,TCO成本降低50%,性能提升数倍。
889 5
|
SQL 数据库连接 数据库
在C++的QT框架中实现SQLite数据库的连接与操作
以上就是在C++的QT框架中实现SQLite数据库的连接与操作的基本步骤。这些步骤包括创建数据库连接、执行SQL命令、处理查询结果和关闭数据库连接。在实际使用中,你可能需要根据具体的需求来修改这些代码。
788 14
(5)Qt中的日期和时间
本文介绍了Qt中处理日期和时间的类QDate、QTime和QDateTime,包括它们的格式化选项、构造函数、公共成员函数以及如何进行日期和时间的增减、比较,同时提到了QElapsedTimer作为QTime的替代品用于计时。
1165 0
|
C# Android开发 iOS开发
9 个 .NET UI 框架,您的选择是?
介绍 9 款 .NET UI 框架,有你的菜麽。
3104 1
9 个 .NET UI 框架,您的选择是?
|
人工智能 Ubuntu C语言
【Ubuntu工具】详细图文教程:Ubuntu系统上安装QT6.2
【Ubuntu工具】详细图文教程:Ubuntu系统上安装QT6.2
2153 0
|
存储 测试技术
西门子S7-200 SMART自由口通信,如何编写接收程序
本篇我们来学习编写西门子S7-200 SMART自由口通信的接收程序。我们继续使用上篇文中STEP7 Micro/WIN SMART编程软件所创建的项目,删除已经编写的发送程序,再来编写接收程序,拖动常开触点到程序段1中,输入地址SM0.1,通过传送指令,将常数十六进制数09送到SMB130。
西门子S7-200 SMART自由口通信,如何编写接收程序
|
安全 程序员 编译器
C++程序中的基类与派生类转换
C++程序中的基类与派生类转换
333 1

热门文章

最新文章

下一篇
开通oss服务