Modbus-RTU详解

news2025/1/9 15:45:43

目录

Modbus-RTU协议

帧结构示例

CRC16校验算法

CRC16算法的过程

modbus-rtu的使用

发送数据

接收数据

tcp网口完整实现modbus-rtu协议

使用NModbus4实现modbus-rtu协议

安装NModbus4库。

串口实现NModbus4


Modbus-RTU协议

Modbus RTU 协议是一种开放的串行协议,广泛应用于当今的工业监控设备中。该协议使用 RS-232 或 RS-485 串行接口进行通信,并得到市场上几乎所有商业 SCADA、HMI、OPC 服务器和数据采集软件程序的支持。因此,很容易将 Modbus 兼容设备集成到新的或现有的监控应用程序中,并具有即时的软件支持。

帧结构:设备地址、功能码、数据和CRC校验字段。

常用功能码:Modbus-RTU协议定义了一系列常用的功能码,用于执行不同的操作,如读取保持寄存器、写入单个寄存器、写入多个寄存器等。

(1). 功能码-0x03读保持寄存器:该功能码用于从设备中读取一个或多个保持寄存器的值。
(2). 功能码-0x06写单个寄存器:该功能码用于向设备中写入一个保持寄存器的值。
(3). 功能码-0x10写多个寄存器:该功能码用于向设备中写入多个连续的保持寄存器的值。

帧结构示例

Modbus RTU 功能 01 用于从 Modbus 从站数据采集设备读取线圈状态或数字输出状态。请参阅下面的典型命令和响应以及使用说明。

    主机发送:  01 03 00 00 00 01 84 0A
    从机响应:  01 03 02 19 98 B2 7E

该例子中,主机发送的数据为`地址 + 功能码 + 数据 + 校验`​,CRC校验码是根据前面的数据计算得出的

回复的数据格式

CRC16校验算法

CRC全称循环冗余校验(Cyclic Redundancy Check, CRC),是通信领域数据传输技术中常用的检错方法,用于保证数据传输的可靠性。

CRC校验的基本思路是数据发送方发送数据之前,先生成一个CRC校验码,可以是单bit也可以是多bit,并附在有效数据末尾,以串行方式发送到接收方。接收方接收到数据后,进行CRC校验,根据校验结果就可以知道数据是否有误。

CRC校验码的生成:将有效数据**扩展后**作为被除数,使用一个指定的**多项式**作为除数,进行模二除法,得到的**余数**就是校验码。

数据接收方的CRC校验:将接受的数据(**有效数据+CRC校验码**)扩展后作为被除数,用指定的多项式作为除数,进行模二除法,得到**余数为0**,则表示校验正确。

我们使用代码向设备发送命令帧时需要使用CRC算法计算校验值,当设备响应数据时使用CRC算法校验该数据是否正确,

CRC16算法的过程

1 初始化一个16位的寄存器地址 用作初始值
2 遍历数据字节,从最高位到最低位, 
3 将数据字节与寄存器异或
4 对寄存器进行8次迭代,每一次迭代将寄存器右移一位
5 如果最低位位1,将寄存器与生成多项式0x8005异或,否则只进行右移操作
6 重复上述步骤直到遍历完所有的字节
7 最终寄存器的值就是crc16校验码
8 crc计算之后高低位进行互换

以下是封装的CRC16效验算法类:

    public static class CRC16
    {
        /// <summary>
        /// CRC校验,参数data为byte数组
        /// </summary>
        /// <param name="data">校验数据,字节数组</param>
        /// <returns>字节0是高8位,字节1是低8位</returns>
        public static byte[] CRCCalc(byte[] data)
        {
            //crc计算赋初始值
            int crc = 0xffff;
            for (int i = 0; i < data.Length; i++)
            {
                crc = crc ^ data[i];
                for (int j = 0; j < 8; j++)
                {
                    int temp;
                    temp = crc & 1;
                    crc = crc >> 1;
                    crc = crc & 0x7fff;
                    if (temp == 1)
                    {
                        crc = crc ^ 0xa001;
                    }
                    crc = crc & 0xffff;
                }
            }
            //CRC寄存器的高低位进行互换
            byte[] crc16 = new byte[2];
            //CRC寄存器的高8位变成低8位,
            crc16[1] = (byte)((crc >> 8) & 0xff);
            //CRC寄存器的低8位变成高8位
            crc16[0] = (byte)(crc & 0xff);
            return crc16;
        }
    
        /// <summary>
        /// CRC校验,参数为空格或逗号间隔的字符串
        /// </summary>
        /// <param name="data">校验数据,逗号或空格间隔的16进制字符串(带有0x或0X也可以),逗号与空格不能混用</param>
        /// <returns>字节0是高8位,字节1是低8位</returns>
        public static byte[] CRCCalc(string data)
        {
            //分隔符是空格还是逗号进行分类,并去除输入字符串中的多余空格
            IEnumerable<string> datac = data.Contains(",") ? data.Replace(" ", "").Replace("0x", "").Replace("0X", "").Trim().Split(',') : data.Replace("0x", "").Replace("0X", "").Split(' ').ToList().Where(u => u != "");
            List<byte> bytedata = new List<byte>();
            foreach (string str in datac)
            {
                bytedata.Add(byte.Parse(str, System.Globalization.NumberStyles.AllowHexSpecifier));
            }
            byte[] crcbuf = bytedata.ToArray();
            //crc计算赋初始值
            return CRCCalc(crcbuf);
        }
    
        /// <summary>
        ///  CRC校验,截取data中的一段进行CRC16校验
        /// </summary>
        /// <param name="data">校验数据,字节数组</param>
        /// <param name="offset">从头开始偏移几个byte</param>
        /// <param name="length">偏移后取几个字节byte</param>
        /// <returns>字节0是高8位,字节1是低8位</returns>
        public static byte[] CRCCalc(byte[] data, int offset, int length)
        {
            byte[] Tdata = data.Skip(offset).Take(length).ToArray();
            return CRCCalc(Tdata);
        }
    }

modbus-rtu的使用

发送数据

现要读取变送器设备(地址 0x01)的风速值,文档如图所示

我们发送的命令帧应为

根据命令帧计算校验码(3种方式)

    byte[] buffer = new byte[] { 0x01, 0x03, 0x00, 0x00, 0x00, 0x01 };
    byte[] crc16 = CRC16.CRCCalc(buffer);  // 根据字节数组计算
    Console.WriteLine($"{crc16[0]:X2} {crc16[1]:X2}");
    
    string data1 = "0x01,0x03,0x00,0x00,0x00,0x02";
    string data1 = "0x01 0x03 0x00 0x00 0x00 0x02";
    byte[] crc16 = CRC16.CRCCalc(data1);    // 根据字符串计算
    Console.WriteLine($"{crc16[0]:X2} {crc16[1]:X2}");
    
     byte[] buffer = new byte[] { 0x01, 0x03, 0x00, 0x00, 0x00, 0x02, 0xC4, 0x0B };
     byte[] crc16 = CRC16Calc(buffer,0,6);  // 从字节数组中截取某部分计算
     Console.WriteLine($"{crc16[0]:X2} {crc16[1]:X2}");

将数据和校验码数组进行合并然后发送

    byte[] buffer = new byte[] { 0x01, 0x03, 0x00, 0x00, 0x00, 0x01 };
    byte[] crc16 = CRC16.CRCCalc(buffer);
    byte[] data = buffer.Concat(crc16).ToArray();
    serialPort.Write(data, 0, data.Length);

接收数据

我们将命令帧(请求帧、问询码)发送后,如果没有错误,从设备会返回对应的数据,如下读取变送器设备(地址 0x01)的实时风力等级值,将会返回如图所示的数据,我们需要将数据读取、校验、计算、展示

    // 假设这是从设备响应的数据
    // 0x01:设备地址码
    // 0x03:功能码
    // 0x02:读取到的数据字节
    // 0x00, 0x01:当前风力等级
    // 0x79, 0x84:校验码
    byte[] value = new byte[] { 0x01, 0x03, 0x02, 0x00, 0x01, 0x79, 0x84 };
    // 1、验证校验码是否正确
    byte[] crc = CRC16.CRCCalc(value, 0, value.Length - 2);
    if (crc[0] != value[value.Length - 2] || crc[1] != value[value.Length-1] ) {
        MessageBox.Show("数据校验错误,应忽略");
        return;
    }
    // 2、验证设备地址
    if (value[0] !=  0x01)
    {
        MessageBox.Show("设备地址不正确");
        return;
    }
    // 3、计算数据
    // int v = value[3] * 256 + value[4];  // 因为这个数据占两个字节,每个字节最大255,相当于256进制,转换为10进制
    int v = (value[3] << 8) + value[4];       // 也可以使用左移运算符,高位左移8位,相当于乘2的8次方
    
    // 4、数据展示
    MessageBox.Show("风力等级:" + v);

tcp网口完整实现modbus-rtu协议

public partial class Form1 : Form
{
    /// <summary>
    /// 套接字
    /// </summary>
    Socket socket;
    
    /// <summary>
    /// IP地址
    /// </summary>
    string Ip = "192.168.107.5";

    /// <summary>
    /// 端口
    /// </summary>
    string Dk = "8016";

    /// <summary>
    /// 命令帧
    /// </summary>
    string Icommand = "01 03 00 00 00 02";

    public Form1()
    {
        InitializeComponent();
        button1.Enabled = false;
        checkBox1.Enabled = false;
    }

    /// <summary>
    /// 打开连接
    /// </summary>
    /// <param name="sender"></param>
    /// <param name="e"></param>
    private void button2_Click(object sender, EventArgs e)
    {
        if (button2.Text == "连接网口")
        {
            try
            {
                socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
                socket.Connect(Ip, int.Parse(Dk));

                button1.Enabled = true;
                checkBox1.Enabled = true;
                button2.Text = "断开";
                this.timer1.Start();
            }
            catch (Exception ex)
            {
                MessageBox.Show(ex.Message);
            }
        }
        else
        {
            if (socket == null) return;
            socket.Close();
            button1.Enabled = false;
            checkBox1.Enabled = false;
            checkBox1.Checked = false;
            button2.Text = "连接网口";
            this.timer1.Stop();
        }
        
    }


    /// <summary>
    /// 刷新风速风向数据
    /// </summary>
    /// <param name="sender"></param>
    /// <param name="e"></param>
    private async void button1_Click(object sender, EventArgs e)
    {
        byte[] bs = new byte[1024];
        bs = StringToByte(Icommand);
        byte[] bb = CRCCalc(bs);
        bs = bs.Concat(bb).ToArray();
        await Task.Run(() =>
        {
            socket.Send(bs); // 发送请求帧

            byte[] body = new byte[1024];
            int length = socket.Receive(body); // 获取响应帧

            double value = (body[3] * 256+ body[4]) *0.01;
            double value2 = (body[5] * 256 + body[6]);
            this.Invoke(new Action(() =>
            {
                this.textBox1.Text = value.ToString() + "m/s";
                this.textBox2.Text = value2.ToString();
            }));

        });

    }



    /// <summary>
    /// 字符串转字节
    /// </summary>
    /// <param name="s"></param>
    /// <returns></returns>
    byte[] StringToByte(string s)
    {
        string[] strings = s.Split(' ') ;
        byte[] bs = new byte[strings.Length];
        for (int i = 0; i < strings.Length; i++)
        {
            bs[i] = Convert.ToByte(strings[i],16);
        }
        return bs;
    }


    /// <summary>
    /// CRC效验
    /// </summary>
    /// <param name="data"></param>
    /// <returns></returns>
    public static byte[] CRCCalc(byte[] data)
    {
        //crc计算赋初始值
        int crc = 0xffff;
        for (int i = 0; i < data.Length; i++)
        {
            //XOR
            //(1) 0^0=0,0^1=1  0异或任何数=任何数
            //(2) 1 ^ 0 = 1,1 ^ 1 = 0  1异或任何数-任何数取反
            //(3) 1 ^ 1 = 0,0 ^ 0 = 0  任何数异或自己=把自己置0
            //异或操作符是^。异或的特点是相同为false,不同为true。
            crc = crc ^ data[i]; //和^表示按位异或运算。
                                 //0x0fff ^ 0x01 Console.WriteLine(result.ToString("X")); 
                                 // 输出结果为4094,即十六进制数1001
            for (int j = 0; j < 8; j++)
            {
                int temp;
                temp = crc & 1; // & 运算符(与) 1 & 0 为 0  ;0 & 0 为0;1 & 1 为1

                //右移 (>>) 将第一个操作数向右移动第二个操作数所指定的位数,空出的位置补0。右移相当于整除. 右移一位相当于除以2;右移两位相当于除以4;右移三位相当于除以8。
                //int i = 7;
                //int j = 2;
                //Console.WriteLine(i >> j);   //输出结果为1
                crc = crc >> 1;
                crc = crc & 0x7fff;
                if (temp == 1)
                {
                    crc = crc ^ 0xa001;
                }
                crc = crc & 0xffff;
            }
        }
        //CRC寄存器的高低位进行互换
        byte[] crc16 = new byte[2];
        //CRC寄存器的高8位变成低8位,
        crc16[1] = (byte)((crc >> 8) & 0xff);
        //CRC寄存器的低8位变成高8位
        crc16[0] = (byte)(crc & 0xff);
        return crc16;
    }
}

使用NModbus4实现modbus-rtu协议

NModbus4是一个C#实现的Modbus库,它允许开发者以Modbus RTU的方式与工业设备进行通信。

安装NModbus4库。

通过NuGet安装NModbus4

串口实现NModbus4

public partial class Form1 : Form
{
    // 创建对象
    ModbusSerialMaster master;
    public Form1()
    {
        InitializeComponent();
        this.serialPort1.PortName = "COM2"; // 串口名
        this.serialPort1.BaudRate = 9600;
        this.serialPort1.DataBits = 8;
        this.serialPort1.Parity =System.IO.Ports.Parity.None;
        this.serialPort1.StopBits = System.IO.Ports.StopBits.One;

        serialPort1.Open();
        master = ModbusSerialMaster.CreateRtu(serialPort1);
    }

    /// <summary>
    ///  读取数据
    /// </summary>
    /// <param name="sender"></param>
    /// <param name="e"></param>
    private async void button1_Click(object sender, EventArgs e)
    {
        // ReadHoldingRegistersAsync 异步读取数据
        // await 等待异步任务执行完之后 再往下执行
        // 参数1 从站地址, 参数2 起始地址  参数3:寄存器个数
        // values 元素个数和寄存器个数有关
        ushort[] values = await master.ReadHoldingRegistersAsync(1,0x00,3);
        comboBox1.DataSource = values;
    }

    /// <summary>
    /// 写入数据
    /// </summary>
    /// <param name="sender"></param>
    /// <param name="e"></param>
    private async void button2_Click(object sender, EventArgs e)
    {
        // 写入 单个的寄存器
        // 参数1 从站地址
        // 参数2 写入的地址
        // 参数3 写入的数据 
        // short 短整型
        // ushort 无符号的短整型
        await master.WriteSingleRegisterAsync(1,0x04,14);
    }
}

本文部分来源网络,如有侵权请联系作者删除!!!

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

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

相关文章

基于51单片机的无线模块PWM电机调速设计

一、概述 为了实现对直流电机无极调速的需求&#xff0c;提出了一种基于STC 89C52微控制器的直流PWM可调速系统设计方案。根据系统所需达到的控制目的&#xff0c;UL2003驱动芯片作为电动机驱动电路&#xff0c;实现对电机的驱动。控制算法采用经典PWM脉宽调制算法作为控制策略…

充电宝哪个牌子好?学生党适合哪种充电宝?推荐四款性价比充电宝

对于学生党而言&#xff0c;保持手机电量充足是学习、社交和娱乐的基本保证。然而&#xff0c;面对频繁的使用&#xff0c;手机电量常常不够用&#xff0c;这时一款性能优良的充电宝就显得尤为重要。那么&#xff0c;充电宝哪个牌子好呢&#xff1f;对于学生党来说&#xff0c;…

番茄钟工作法

目录 1.使用番茄钟的注意事项和技巧: 2.番茄工作法的优点: 3.番茄钟案例: 从棉花糖实验说起 我得了什么「病」&#xff1f; 外界的诱惑 失效的 Deadline 永远停留在纸上的计划 番茄土豆大作战&#xff1a;番茄工作法简明教程 计划 执行 记录与分析 番茄工作法怎么…

可视化图表与源代码显示的动态调整

可视化图表与源代码显示的动态调整 页面效果描述&#xff1a;本篇代码实现了通过拖动一个可调整大小的分隔符&#xff0c;用户可以动态地调整图表显示区域和源代码显示区域的大小。通过监听鼠标事件&#xff0c;当用户拖动分隔符时&#xff0c;会动态计算并更新两个区域的大小 …

俄组织Fighting Ursa利用虚假汽车销售广告传播HeadLace后门

最近&#xff0c;Palo Alto Networks的科研人员揭露了有一个与俄罗斯有关联的威胁行动者——Fighting Ursa&#xff08;亦称APT28、Fancy Bear或Sofacy&#xff09;。该组织通过散布虚假的汽车销售广告&#xff0c;特别是针对外交官群体&#xff0c;散播名为HeadLace的后门恶意…

6款打印刻录监控与审计系统 | 一键解锁器功能探析

信息高度敏感的社会环境&#xff0c;企事业单位对于文档的安全传输、打印与刻录过程的监控与审计需求日益迫切。 然而&#xff0c;为了全面满足读者对安全工具的了解需求&#xff0c;这篇文章小编将首先概述几款领先的打印刻录监控与审计系统&#xff0c;随后简要提及“一键解…

【Java】Collection中自定义类重写contains方法。

如果集合中存储的是自定义对象&#xff0c;也想使用contaisn方法来判断是否包含&#xff0c;那么在javabean类中&#xff0c;一定要重写equals方法。 因为contains方法的底次是使用equals方法实现的&#xff0c;所以重写equals方法。 Main类&#xff1a; package demo;import…

SQL注入(闯关游戏)

目录 关卡1 关卡2 关卡3 关卡4 关卡5 关卡6 关卡7 关卡8 关卡9 关卡10 关卡11 关卡12 关卡13 关卡14 关卡15 关卡16 关卡17 关卡18 关卡19 关卡20 关卡21 关卡22 关卡23 关卡24 关卡1 (联合查询) ?gid1 第一件事情就是逃脱单引号的控制——》为了闭…

vue+element 根据父容器动态设置table高度出滚动条

可以通过CSS样式来控制表格的高度&#xff0c;并使用JavaScript动态地设置这个高度。 HTML: <template><el-table:data"tableData":height"tableHeight"style"width: 100%"><!-- 列配置 --></el-table> </template&…

【Kubernetes】Deployment 的清理策略

Deployment 的清理策略 在 Deployment 中配置 spec.revisionHistoryLimit 字段&#xff0c;可以指定其 清理策略。该字段用于指定 Deployment 保留旧 ReplicaSet 的个数&#xff0c;即更新 Pod 前的版本个数。该字段的默认值是 10。 创建 revisionhistory-demo.yaml 文件&…

一文了解K8S(Kubernates)

一、K8S 1. 概述 Kubernetes 是一个可移植、可扩展的开源平台&#xff0c;用于管理容器化的工作负载和服务&#xff0c;可促进声明式配置和自动化。 Kubernetes 拥有一个庞大且快速增长的生态&#xff0c;其服务、支持和工具的使用范围相当广泛。 Kubernetes 这个名字源于希腊…

Unity 功能 之 创建 【Unity Package】 Manager 自定义管理的包使用配置URL,使用 git URL加载的简单整理

Unity 功能 之 创建 【Unity Package】 Manager 自定义管理的包使用配置URL&#xff0c;使用 git URL加载的简单整理 目录 Unity 功能 之 创建 【Unity Package】 Manager 自定义管理的包使用配置URL&#xff0c;使用 git URL加载的简单整理 一、简单介绍 二、Unity Package …

【你也能从零基础学会网站开发】SQL Server 2000中的数据类型之String字符串类型

&#x1f680; 个人主页 极客小俊 ✍&#x1f3fb; 作者简介&#xff1a;程序猿、设计师、技术分享 &#x1f40b; 希望大家多多支持, 我们一起学习和进步&#xff01; &#x1f3c5; 欢迎评论 ❤️点赞&#x1f4ac;评论 &#x1f4c2;收藏 &#x1f4c2;加关注 SQL Server 中…

c++ | vector

前言 本篇博客讲解cSTL中的vector &#x1f493; 个人主页&#xff1a;普通young man-CSDN博客 ⏩ 文章专栏&#xff1a;C_普通young man的博客-CSDN博客 ⏩ 本人giee: 普通小青年 (pu-tong-young-man) - Gitee.com 若有问题 评论区见&#x1f4dd; &#x1f389;欢迎大家点赞…

珠江电缆,顺应全球变化,实现高质量出海

在全球经济快速变化的今天&#xff0c;越来越多的企业将目光投向了国际市场。特别是对于线缆行业来说&#xff0c;顺应全球变化、应对机遇与挑战&#xff0c;实现高质量出海已成为长期发展的战略目标之一。珠江电缆作为一家集研发、制造和销售为一体的大型专业电线电缆企业&…

智能电表在什么情况下需要加装互感器?

智能电表作为现代电力系统中不可或缺的一部分&#xff0c;负责准确计量电力消耗。然而&#xff0c;在某些特定条件下&#xff0c;仅凭智能电表无法满足高精度测量需求&#xff0c;此时便需引入互感器。本文将深入解析智能电表与互感器的协作原理&#xff0c;明确指出加装互感器…

ASR(Automatic Speech Recognition)调研,当前sota的架构

asr概览英文纯享版&#xff1a;an overview of transducer models for asr 本文主要讲述nvidia和openai的模型架构&#xff0c;应为他们两家霸榜huggingface leader board 小白也能阅读了解一下当前sota的asr架构是什么样的 评测指标 Word Error Rate (WER)&#xff1a;错词率…

关于使用QListView向模型插入第一条数据的相关问题

在使用QListView的时候&#xff0c;看着它的四个尖角不爽&#xff0c;想着让它变成圆角矩形吗&#xff0c;但麻烦就接踵而至了。。。 在向其中插入第一条数据的时候发现&#xff0c;你插入的一条数据它存在边界超过QListView的现象。如下图所示&#xff1a; 这就令人头大了&am…

JavaEE 图书管理系统

基于阿里巴巴的fastjson框架搭建的JavaEE版本的图书管理系统&#xff0c;项目架构如下&#xff1a; fastjson包的阿里云下载镜像如下&#xff1a; Central Repository: com/alibaba/fastjson2/fastjson2/2.0.8 运行效果&#xff1a; Bean Book.java package Bean;public c…

2024华数杯大学生数学建模竞赛(C题)数学建模完整思路+完整代码全解全析

你是否在寻找数学建模比赛的突破点&#xff1f;数学建模进阶思路&#xff01; 作为经验丰富的数学建模团队&#xff0c;我们将为你带来2024华数杯数学建模竞赛&#xff08;C题&#xff09;的全面解析。这个解决方案包不仅包括完整的代码实现&#xff0c;还有详尽的建模过程和解…