网络编程之Modbus与HTTP

发布于:2025-06-13 ⋅ 阅读:(16) ⋅ 点赞:(0)

Modbus

一、基础概念

1.  起源

● Modbus由Modicon公司于1979年开发,是全球第一个真正用于工业现场的总线协议

● 在中国,Modbus 已经成为国家标准,并有专业的规范文档,感兴趣的可以去查阅相关的文件,详情如下:标准编号为:GB/T19582-2008文件名称:《基于 Modbus 协议的工业自动化网络规范》

● Modbus通信协议具有多个变种,其中有支持串口,以太网多个版本,其中最著名的是Modbus RTU、Modbus ASCII和Modbus TCP三种,其中Modbus TCP是在施耐德收购Modicon后1997年发布的。

2.分类

1)Modbus RTU

运行在串口上的协议,采用二进制表现形式以及紧凑的数据结构,通信效率较高,应用比较广泛

2)Modbus ASCII

运行在串口上的协议,采用ASCII码进行传输,并且每个字节的开始和结束都有特殊字符作为标志,传输效率远远低于Modbus RTU,一般只有通讯量比较少时才会考虑它。

3)Modbus TCP

是一种基于以太网的协议,使用 TCP/IP 协议栈进行通信。它使用以太网帧作为数据传输的封装,通过 IP 地址和端口号来标识设备

3.  优势

简单、免费、容易使用

4.  应用场景

Modbus协议是现在国内工业领域应用最多的协议,不只PLC设备,各种终端设备,比如水控机、水表、电表、工业秤、各种采集设备

 二、Modbus TCP

1.特点

1.  遵循主从问答协议 (主机 从机 主从问答:采集信息 、控制

             (主机问---从机答247      1-247:从机       0:广播      248-255:保留 ) 

2.  Modbus TCP协议是应用层协议,基于传输层TCP通信

3.  Modbus TCP的默认端口号是502

2.组成

Modbus TCP协议包含:报文头、功能码、数据

报文头有7个字节,功能码有1个字节,Modbus TCP协议最大数据帧长度为260个字节,数据最多为252个字节。

2.1报文头(7字节)

事务处理标识符:数据包的标识,一般无限制,主机发啥,从机回啥

协议标识符:0x0000 

长度接下来的字节长度,字段值注意使用十六进制

                                      

单元标识符:从机ID

补充:寄存器
Modbus TCP通过寄存器的方式存储数据。

一共有四种类型的寄存器,分别是:离散量输入、线圈、输入寄存器、保持寄存器。

1) 离散量和线圈其实就是位寄存器(每个寄存器数据占1字节),工业上主要用于控制IO设备。

线圈寄存器,类比为开关量,每一个bit都对应一个信号的开关状态。所以一个byte就可以同时控制8路的信号。比如控制外部8路io的高低。 线圈寄存器支持读也支持写,写在功能码里面又分为写单个线圈寄存器和写多个线圈寄存器。

对应的功能码也就是:0x01 0x05 0x0f

离散输入寄存器,离散输入寄存器就相当于线圈寄存器的只读模式,他也是每个bit表示一个开关量,而他的开关量只能读取输入的开关信号,是不能够写的。比如我读取外部按键的按下还是松开。

所以功能码也简单就一个读的 0x02

2) 输入和保持寄存器是字寄存器(每个寄存器数据占2个字节),工业上主要用于存储工业设备的值。

保持寄存器,这个寄存器的单位不再是bit而是两个byte,也就是可以存放具体的数据量的,并且是可读写的。比如我设置时间年月日,不但可以写也可以读出来现在的时间。写也分为单个写和多个写

所以功能码有对应的三个:0x03 0x06 0x10

输入寄存器,这个和保持寄存器类似,但是也是只支持读而不能写。一个寄存器也是占据两个byte的空间。类比我通过读取输入寄存器获取现在的AD采集值

对应的功能码也就一个 0x04

2.2常用功能码 

寄存器PLC地址和寄存器的对应关系:

线圈(可读可写): 00001-09999    读: 01  写单个: 05    写多个: 0f

离散量输入(只读):10001-19999    读:02    

保持寄存器(可读可写):40001-49999   读:03   写单个:06  写多个:10

输入寄存器(只读):30001-39999 0   读: 04

例:

总结:

读数据:

主机-->从机

线圈、离散输入:报文头 + 功能码 + 起始地址(2)+要读取的线圈数量(2)

保持寄存器、输入寄存器:报文头 + 功能码 + 起始地址(2)+要读取的寄存器数量(2)

从机-->主机

报文头 + 功能码 + 字节计数(1)+ data(不定)

写单个

主机-->从机

线圈:报文头 + 功能码 + 起始地址(2)+ 断通标志(2)

保持寄存器:报文头 + 功能码 + 起始地址(2)+data(2)

从机-->主机

原路返回

写多个

主机-->从机

线圈:报文头 + 功能码 + 起始地址(2)+线圈数量(2)+字节计数(1)+ data(不定)

保持寄存器:报文头 + 功能码 + 起始地址(2)+寄存器数量(2)+字节计数(1)+ data(不定)

从机-->主机

 报文头 + 功能码 +起始地址(2)+存储器数量(2) ---》(取部分原文返回)

3、linux下编程

三、Modbus RTU

 1.与Modbus TCP的区别

在一般工业场景使用modbus RTU的场景还是更多一些,modbus RTU基于串行协议进行收发数据,包括RS232/485等工业总线协议。

与modbus TCP不同的是RTU没有报文头MBAP字段,但是在尾部增加了两个CRC检验字节(CRC16),因为网络协议中自带校验,所以在TCP协议中不需要使用CRC校验码。

RTU和TCP的总体使用方法基本一致,只是在创建modbus对象时有所不同,TCP需要传入网络socket信息;而RTU需要传入串口相关信息。

 2.特点

(1)遵循主从问答的通信方式

(2)采用串口进行通信

 设置串口参数时要求:

波特率为9600(波特率是指每秒钟传输的比特数)

8位数据位 (数据位是指每个字符中包含的比特数)

1位停止位 (停止位是指在每个字符传输结束后添加的比特数)

无流控 (流控是指在数据传输过程中控制数据流量的一种机制,无流控表示在该设置下没有额外的控制机制来控制数据流量)

3.Modbus RTU协议格式

地址码(1) :从机ID

功能码(1) :同Modbus TCP (01 02 03 04 05 06 0f 10H)

数据 :

校验码(2):对地址码、功能码、数据码进行校验,由函数生成

例:

主机-->从机: 01 03 00 00 00 01 84 0a

01 :从机id

03 :功能码

00 00 :起始地址

00 01 :数量

84 0a:校验码

从机-->主机01 03 02 0014 b8 44

01 :从机id

03 :功能码

02 :字节计数

00 14 :数据

b8 44 :校验码

值得收藏 Modbus RTU 协议详解-CSDN博客

四、Modbus库

1.  编程步骤

1.  创建实例

2.  设置从机ID

3.  建立连接

4.  寄存器操作按需选择)

5.  关闭套接字

6.  释放实例

#include <modbus.h>
#include <stdio.h>
//  $$ 客户端 $$  



int main(int argc, char const *argv[])
{

    
    //创建modbus实例
    modbus_t* ctx;
    ctx=modbus_new_tcp(argv[2],atoi(argv[1]));
    if(ctx==NULL)
    {
        perror("modbus_new_tcp err");
        return -1;
    }
    else
    {
        printf("创建实例成功\n");
    }
    //设置从机ID
    if(modbus_set_slave(ctx,1)<0)
    {
        perror("modbus_set_slave err");
        return -1;
    }
    else
    {
        printf("从机ID设置成功\n");
    }
    //建立连接
    if( modbus_connect(ctx)<0)
    {
        perror("modbus_connect err");
        return -1;
    }
    else
    {
        printf("连接成功\n");
    }
   
    uint16_t data[128] = {0};
    //执行03功能
    int num=modbus_read_registers(ctx,0,2,data);
    for (int i = 0; i < num; i++)
    {
        printf("%d ", data[i]);
    }
    putchar(10);
    //关闭套接字
    modbus_close(ctx);
    //关闭连接
    modbus_free(ctx);
    return 0;
}

HTTP协议

1.  http简介

HTTP协议是Hyper Text Transfer Protocol(超文本传输协议)的缩写,是用于Web Browser(浏览器)到Web Server(服务器)进行数据交互的传输协议。

HTTP是应用层协议

HTTP是一个基于TCP通信协议传输来传递数据(HTML 文件, 图片文件, 查询结果等)

HTTP协议工作于B/S架构上,浏览器作为HTTP客户端通过URL主动向HTTP服务端即WEB服务器发送所有请求,Web服务器根据接收到的请求后,向客户端发送响应信息。

HTTP默认端口号为80,但是你也可以改为其他端口

2.  http特点

HTTP是无连接:无连接的含义是限制每次连接只处理一个请求。服务器处理完客户的请求,并给客户应答后,即断开连接。采用这种方式可以节省传输时间。

(需要注意一点:HTTP协议本身是无连接的,即每个请求和响应都是独立的。但是http是基于TCP协议的连接管理方式,长连接和短连接用于优化HTTP请求和响应的传输效率。长连接是指在一个TCP连接上可以发送多个HTTP请求和响应,而不需要每次请求都建立和关闭一个新的TCP连接。短连接是指每个HTTP请求和响应都使用一个新的TCP连接。)

HTTP是媒体独立:这意味着,只要客户端和服务器知道如何处理的数据内容,任何类型的数据都可以通过HTTP发送。客户端以及服务器指定使用适合的MIME-type内容类型。

HTTP是无状态:HTTP协议是无状态协议。无状态是指协议对于事务处理没有记忆能力。无状态意味着如果后续处理需要前面的信息,则它必须重传,这样可能导致每次连接传送的数据量增大。另一方面,在服务器不需要先前信息时它的应答就较快。

HTTP 协议同学可能都知道,HTTP 协议是以明文方式发送内容,不提供任何方式的数据加密,如果攻击者截取了 Web 浏览器和网站服务器之间的传输报文,就可以直接读懂其中的信息,因此 HTTP 协议不适合传输一些敏感信息,比如信用开号、密码等。
为了解决 HTTP 协议的这一缺陷,需要使用另一种协议:HTTPS 协议。
HTTPS(全称:Hyper Text Transfer Protocol over Secure Socket Layer),是以安全为目标的 HTTP 通道,简单来说就是 HTTP 的安全版。即在 HTTP 下加入 SSL 协议,SSL 依靠证书来验证服务器的身份,并为浏览器和服务器之间的通信加密。
HTTPS 和 HTTP 的区别主要为以下三点:
● 1.http 是超文本传输协议,信息是明文传输;https 协议是由 http + ssl 协议构建的可进行加密传输、身份认证的网络协议,信息是密文传输,比 http 协议安全。
● 2.https 协议需要到 ca 申请证书,一般免费证书很少,需要缴费
● 3.http 和 https 使用的默认端口也不一样,前者是 80,后者是 443

3.http协议格式

1)客户端请求数据格式

客户端发送一个HTTP请求到服务器的请求消息包括以下格式:请求行、请求头部、空行和请求数据四个部分组成,下图给出了请求报文的一般格式。

注意:linux下回车符和换行符相同

URL:路径

请求方法:http协议中共定义了八种数据的请求方法。分别是:OPTIONS、HEAD、GET、POST、PUT、DELETE、TRACE、CONNECT;我们在实际应用中常用的也就是 get 和 post,其他请求方式也都可以通过这两种方式间接的来实现。

2)服务器响应数据格式

HTTP响应也由四个部分组成,分别是:状态行、消息报头、空行和响应正文

状态行:由三部分组成,HTTP协议的版本号、状态码、以及对状态码的文本描述。例如:HTTP/1.1 (协议版本)200 (状态码)OK (CRLF) 。(200表示请求已经成功)