最近在做采集的一些任务所以学了一下Modbus通信,学了好几天昨天终于把大概弄明白了,其实简单来说就是客户端向设备发送一个请求报文请求数据,服务器端根据请求报文向客户端端回发一个报文,客户端在接收到响应报文之后对响应报文进行解析,解析之后在将数据存到数据库里或者打印出来。
报文格式:
下面让我们来详细的了解一下这个过程。
首先,做就是在客户端添加一个SOCKET与设备建立连接。说简单点,你不要认为是什么Modbus,就认为是一台电脑,开了SOCKET服务在这里。参考代码如下:
public static void testframe () { TcpClient client = new TcpClient (); client.Connect (IPAddress.Parse ("127.0.0.1"), 5556); }
这里IP和端口号都是由设备方提供的。真实程序一般都把这两个参数写配置文件中。
设备连上以后,下一步当然就是读取数据。Modbus的基本原理就是程序向设备请求,需要读取哪个数据,设备就会返回相应的数据。我们知道机器或者说是电脑是只认识01001这样的字符串的。所以所谓的Modbus协议,说得简单一点,就是规定这样一个0101字符各代表什么含义。
下面是核心代码即请求报文和响应报文的代码:
if (client.Connected) { NetworkStream sm = client.GetStream (); // start address = 99 , count = 3 sm.Write (new byte[] { 0x01, 0x00, // transfer flag (little-endian) 0x00, 0x00, // protocol flag 0x00, 0x06, // length (big-endian) 0x01, // device id 0x03, // function code 0x00, 0x63, // start address (big-endian) 0x00, 0x03 // count (big-endian) }, 0, 12); byte [] frame = new byte [256]; int read_len = sm.Read (frame, 0, frame.Length); if (read_len > 9) { Console.WriteLine ("successful in receiving:"); int cnt = (int) frame [8]; for (int i = 0; i < cnt; i += 2) { uint val = (uint) frame [i + 9]; val <<= 8; val |= (uint) frame[i + 10]; Console.WriteLine (val); } } else if (read_len == 9) { Console.WriteLine ("rcv error!!"); Console.WriteLine ("error code : " + (frame[7] - 0x80) + frame[8]); } }
下面对这段代码进行一下解释:
MSDN里对NetworkStream 类的解释是“提供用于网络访问的基础数据流。”,并且“要创建 NetworkStream,必须提供连接的Socket。”。
NetworkStream.Write 方法 //将数据写入 NetworkStream。(发送请求报文) ,方法里有三个参数分别是
byte [] frame = new byte [256]; //定义一个标识名为frame的byte类型的数组用于存放接收到的数据流 int read_len = sm.Read (frame, 0, frame.Length); //NetworkStream.Read的返回值是<span><span><span class="sentence" id="mt5">从 <span><a target=_blank href="https://msdn.microsoft.com/zh-cn/library/system.net.sockets.networkstream.aspx" target="_blank">NetworkStream</a></span> 中读取的字节数。</span></span></span> if (read_len > 9) { Console.WriteLine ("successful in receiving:"); int cnt = (int) frame [8]; //响应报文的第9个byte表示的是数据部分的长度 for (int i = 0; i < cnt; i += 2) { //i+=2是因为每个数据是2byte uint val = (uint) frame [i + 9]; //一个数据是2byte,其中前一个byte是高位,后一个byte是低位 val <<= 8; //左移8位将高位的数放入高位 val |= (uint) frame[i + 10]; //微运算,将低位的数与高位的数取或 Console.WriteLine (val); } }