MODBUS RTU/TCP通讯实例C#(附源码)

根据对MODBUS TCP跟RTU之间通讯协议封装后写了一个通讯的DEMO例子,界面是参考的HSL软件的布局,主要是针对线圈寄存器和保存寄存器的多读写。
在这里插入图片描述
主要的功能:

  1. MODBUS TCP/RUT两种通讯方式
  2. BOOL类型线圈寄存器的多读写
  3. 保存寄存器各种数据类型的写入读取
  4. 通讯时发送和读取到的字节数据日志

主要逻辑:
首先是跟串口或者TCP通讯,
其次是数据类型的写入,根据不同数据类型的字节长度读取跟写入对应的寄存器长度。
目前主要是针对数据的写入跟读取做一个简单的DEMO,扩展可以在这个之上进行扩展业务逻辑
部分代码展示

	public static RtuHelper Intance = new RtuHelper();
		//实例化串口
		public static SerialPort port = new SerialPort();
		//实例SOCKET
		public Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp) { SendTimeout = 300, ReceiveTimeout = 1000 };

		private Int16 SendNum = 0;
		//委托
		public delegate void ReceiveData(object str);

		public static event ReceiveData DataLog;
		//打开串口配置串口
		public static void SerialClass(ComClass p)
		{
			port.PortName = p.PortName;
			port.BaudRate = p.BaudRate;
			port.DataBits = p.Databit;
			switch (p.Stopbit)
			{
				case 1:
					port.StopBits = StopBits.One; //使用一个停止位
					break;
				case 1.5f:
					port.StopBits = StopBits.OnePointFive; //使用一个停止位
					break;
				case 2:
					port.StopBits = StopBits.Two; //使用一个停止位
					break;
				default:

					break;
			}
			switch (p.Paritybit)
			{
				case "NONE":
					port.Parity = Parity.None;
					break;
				case "ODD":
					port.Parity = Parity.Odd;
					break;
				case "EVEN":
					port.Parity = Parity.Even;
					break;
				case "MARK":
					port.Parity = Parity.Mark;
					break;
				case "SPACE":
					port.Parity = Parity.Space;
					break;
				default:

					break;
			}
			//port.Handshake = Handshake.None;
			port.RtsEnable = true;
			port.DtrEnable = true;//获取或设置一个值,改值在串行通讯过程中启用数据终端就绪(DTR)信号。
			port.ReadTimeout = 2000;
		}
		/// <summary>
		/// 打开串口
		/// </summary>
		/// <returns></returns>
		public static bool OpenPart()
		{
			bool ok = false;
			//如果串口是打开的,先关闭
			if (port.IsOpen) port.Close();
			try
			{
				//打开串口
				port.Open();
				ok = true;
			}
			catch (Exception Ex)
			{
				throw Ex;
			}
			return ok;
		}
		/// <summary>
		/// 关闭串口
		/// </summary>
		public static void closePort()
		{
			if (port.IsOpen)
			{
				port.Close();
			}
		}
		//关闭socket通讯
		public void DisConnect()
		{
			socket.Disconnect(true);
		}
		//打开Socket通讯
		public bool ReConnect(string IP, int Port)
		{
			try { IPAddress.Parse(IP); }
			catch { return false; }
			socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp) { SendTimeout = 300, ReceiveTimeout = 1000 };
			socket.Connect(IPAddress.Parse(IP), Port);
			// task.Wait(ConnTimeout);
			return socket.Connected;
		}

		public List<bool> ReadBools(string addr, UInt16 count, ModbusType modbusType)
		{
			List<Boolean> rst = new List<Boolean>();
			if (!Read<Boolean>(addr, count, ref rst, modbusType))
				Console.WriteLine("ReadBools");
			return rst;
		}
		public bool WriteBools(string addr, UInt16 count, List<bool> value, ModbusType modbusType)
		{
			return (Write<Boolean>(addr, count, value, modbusType));
		}
	private bool Read<T>(string addr, UInt16 count, ref List<T> rst, ModbusType modbusType)
		{
			//byte[]需要指定长度,不支持Linq
			List<byte> command = new List<byte>();
			ParseReadSend(addr, count, rst, ref command, modbusType);
			byte[] Read = new byte[1024 * 1024];
			int len = 0;
			if (modbusType == ModbusType.RTU)
			{
				if (!SendRTU(command.ToArray(), ref Read))
					return false;
				//报文组成完成
			}
			else if (modbusType == ModbusType.TCP)
			{
				if (!SocketSend(command.ToArray(), ref len, ref Read))
					return false;
			}
			if (!AnalysisData(Read, ref rst, count, modbusType))
				return false;
			return true;
		}
		private bool Write<T>(string addr, UInt16 count, List<T> rst, ModbusType modbusType)
		{
			//byte[]需要指定长度,不支持Linq
			List<byte> command = new List<byte>();
			ParseWriteSend(addr, count, rst, ref command, modbusType);
			byte[] Read = new byte[1024 * 1024];
			int len = 0;
			if (modbusType == ModbusType.RTU)
			{
				if (!SendRTU(command.ToArray(), ref Read))
					return false;
			}
			else if (modbusType == ModbusType.TCP)
			{
				if (!SocketSend(command.ToArray(), ref len, ref Read))
					return false;
			}
			return true;
		}
		private bool SocketSend(byte[] array, ref int recvlen, ref byte[] ReadData)
		{
			if (!socket.Connected)
				return false;
			if (array.Length != socket.Send(array))
				return false;
			DataLog($"[{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")} ModbusTcp]:TCP 发送:" + BitConverter.ToString(array).Replace('-', ' ') + "\t\r");
			recvlen = socket.Receive(ReadData);
			ReadData = ReadData.Skip(0).Take(recvlen).ToArray();
			DataLog($"[{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")} ModbusTcp]:TCP 接收:" + BitConverter.ToString(ReadData).Replace('-', ' ') + "\t\r");
			SendNum++;
			return true;
		}
		private bool SendRTU(byte[] SendData, ref byte[] ReadData)
		{
			if (!port.IsOpen)
				return false;
			port.Write(SendData, 0, SendData.Length);
			DataLog($"[{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")} ModbusTcp]:Rtu 发送:" + BitConverter.ToString(SendData).Replace('-', ' ') + "\t\r");
			Thread.Sleep(500);
			//响应报文接收和解析
			byte[] read = new byte[port.BytesToRead];
			port.Read(read, 0, read.Length);
			DataLog($"[{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")} ModbusTcp]:Rtu 接收:" + BitConverter.ToString(read).Replace('-', ' ') + "\t\r");
			ReadData = read;
			return true;
		}
		//CRC校验
		static List<byte> CRC16(List<byte> value, ushort poly = 0xA001, ushort crcInit = 0xFFFF)
		{
			if (value == null || !value.Any())
			{
				throw new ArgumentException("");
			}
			//计算
			ushort crc = crcInit;
			for (int i = 0; i < value.Count; i++)
			{
				crc = (ushort)(crc ^ (value[i]));
				for (int j = 0; j < 8; j++)
				{
					crc = (crc & 1) != 0 ? (ushort)((crc >> 1) ^ poly) : (ushort)(crc >> 1);
				}
			}
			byte hi = (byte)((crc & 0xFF00) >> 8);//高位置
			byte lo = (byte)(crc & 0x00FF);//底位置
			List<byte> buffer = new List<byte>();
			buffer.AddRange(value);
			buffer.Add(lo);
			buffer.Add(hi);
			return buffer;
		}
  • 13
    点赞
  • 61
    收藏
    觉得还不错? 一键收藏
  • 14
    评论
* * MODBUS协议 * * * 介绍: * 此modbus上位机 协议类 具有较强的通用性 * 本协议类最主要的思想是 把所有向下位机发送的指令 先存放在缓冲区中(命名为管道) * 再将管道中的指令逐个发送出去。 * 管道遵守FIFO的模式。管道中所存放指令的个数 在全局变量中定义。 * 管道内主要分为两部分:1,定时循环发送指令。2,一次性发送指令。 * 定时循环发送指令:周期性间隔时间发送指令,一般针对“输入寄存器”或“输入线圈”等实时更新的变量。 * 这两部分的长度由用户所添加指令个数决定(所以自由性强)。 * 指令的最大发送次数,及管道中最大存放指令的个数在常量定义中 可进行设定。 * * 使用说明: * 1,首先对所定义的寄存器或线圈进行分组定义,并定义首地址。 * 2,在MBDataTable数组中添加寄存器或线圈所对应的地址。 注意 寄存器:ob = new UInt16()。线圈:ob = new byte()。 * 3,对所定义的地址 用属性进行定义 以方便在类外进行访问及了解所对应地址的含义。 * 4,GetAddressValueLength函数中 对使用说明的"第一步"分组 的元素个数进行指定。 * 5,在主程序中调用MBConfig进行协议初始化(初始化内容参考函数)。 * 6,在串口中断函数中调用MBDataReceive()。 * 7,定时器调用MBRefresh()。(10ms以下) * 指令发送间隔时间等于实时器乘以10。 例:定时器5ms调用一次 指令发送间隔为50ms。 * 8,在主程序初始化中添加固定实时发送的指令操作 用MBAddRepeatCmd函数。 * 9,在主程序运行过程中 根据需要添加 单个的指令操作(非固定重复发送的指令)用MBAddCmd函数。
modbus是一种通信协议,常用于工业自动化领域中的设备之间的数据传输。在实际应用中,可以使用C语言编写程序来实现modbus RTUmodbus TCP通信。 modbus RTU通信示例中,可以先设置串口参数,如波特率、数据位等。然后使用串口读写函数来与modbus设备进行通信。例如,可以使用C语言中的open、read和write函数来打开串口、读取数据和写入数据。 以下是一个modbus RTU通信实例的C代码: ``` #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <fcntl.h> #include <termios.h> int main() { int fd; struct termios serial; fd = open("/dev/ttyS0", O_RDWR | O_NOCTTY); if (fd == -1) { perror("open"); return -1; } memset(&serial, 0, sizeof(serial)); serial.c_iflag = 0; serial.c_oflag = 0; serial.c_cflag = CS8 | CREAD | CLOCAL; serial.c_lflag = 0; serial.c_cc[VMIN] = 1; serial.c_cc[VTIME] = 5; cfsetospeed(&serial, B9600); cfsetispeed(&serial, B9600); if (tcsetattr(fd, TCSANOW, &serial) == -1) { perror("tcsetattr"); return -1; } // 使用read和write函数与modbus设备进行通信,具体的读写操作需要根据modbus协议进行解析和封装。 // ... close(fd); return 0; } ``` 而modbus TCP通信则是在基于TCP/IP的网络中实现的,需要使用套接字编程来实现。以下是一个modbus TCP通信实例的C代码: ``` #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <unistd.h> int main() { int sockfd; struct sockaddr_in serveraddr; sockfd = socket(AF_INET, SOCK_STREAM, 0); if (sockfd == -1) { perror("socket"); return -1; } memset(&serveraddr, 0, sizeof(serveraddr)); serveraddr.sin_family = AF_INET; serveraddr.sin_port = htons(502); inet_pton(AF_INET, "192.168.1.1", &(serveraddr.sin_addr)); if (connect(sockfd, (struct sockaddr *)&serveraddr, sizeof(serveraddr)) == -1) { perror("connect"); return -1; } // 使用send和recv函数与modbus设备进行通信,具体的读写操作需要根据modbus协议进行解析和封装。 // ... close(sockfd); return 0; } ``` 以上是实现modbus RTUmodbus TCP通信的简单示例,具体的读写操作和数据解析需要根据实际需求和modbus协议进行相应的处理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 14
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值