ModbusRTU协议的Java Socket

昨天发过来一个WinTcpS7_1K.dll文件,在VB代码里跑并发数据到Modbus Slave,发现当Modbus Slave选择用“Modbus RTU over TCP/IP”连接时有数据传输过来,因此可以确定VB代码使用该DLL跟下位机进行通信时用的是ModbusRTU协议。

因此我就去模拟了直接用socket跟Modbus Slave进行ModbusRTU的通信,先给出ModbusRTU发送协议说明:


测试的是读取保持寄存器的内容,所以功能码为3;数据地址为16位,表示从哪里开始读取数据(可以理解成偏移量);数据长度也为16位,表示读取的数据量大小;最后的16位为CRC-16 Modbus校验,根据前面的内容通过校验算法得到。

测试发送的数据为:byte[] sendInfo = new byte[] {0x01, 0x03, 0x00, 0x00, 0x00, 0x0a, (byte) 0xc5, (byte) 0xcd},表示从偏移量为0的地址开始,顺序读取保持寄存器的10个数据,其中校验码的算法如下:

public static String getCrc16(byte[] arr_buff) {
	int len = arr_buff.length;
	//预置 1 个 16 位的寄存器为十六进制FFFF, 称此寄存器为 CRC寄存器。
	int crc = 0xFFFF;
	int i, j; 
	for (i = 0; i < len; i++) {
		//把第一个 8 位二进制数据 与 16 位的 CRC寄存器的低 8 位相异或, 把结果放于 CRC寄存器
		crc = ((crc & 0xFF00) | (crc & 0x00FF) ^ (arr_buff[i] & 0xFF));
		for (j = 0; j < 8; j++) {
			//把 CRC 寄存器的内容右移一位( 朝低位)用 0 填补最高位, 并检查右移后的移出位
			if ((crc & 0x0001) > 0) {
				//如果移出位为 1, CRC寄存器与多项式A001进行异或
				crc = crc >> 1;
				crc = crc ^ 0xA001;
			} else
				//如果移出位为 0,再次右移一位
				crc = crc >> 1;
		}
	}
	return Integer.toHexString(crc);
}
现在Modbus Slave当中有数据:

运行代码后,得到的返回结果为:01 03 14 0001 0002 0003 0004 0005 0006 0007 0008 0009 000A 8F16

其中的01、03跟发送的数据是一样的内容,14表示返回的查询数据长度(0x14转换为十进制为20,即两倍数据量大小的长度),最后的8F16为前面所有内容的CRC-16 Modbus校验码。

上面就是测试ModbusRTU的内容,还没有测试读取输入状态的内容(即功能码为02),之后将继续测试。

Modbus RTU是一种常用的串行通信协议,用于在不同设备之间进行数据传输。在Java中实现Modbus RTU协议需要使用串口通信库和相关的Modbus库。以下是一个简单的Java代码示例,演示如何使用jSerialComm库实现Modbus RTU协议。 首先,确保你已经安装了jSerialComm库。你可以在Maven中添加以下依赖项: ```xml <dependency> <groupId>com.fazecast</groupId> <artifactId>jSerialComm</artifactId> <version>2.6.2</version> </dependency> ``` 然后,你可以使用以下代码来实现Modbus RTU协议的读取和写入操作: ```java import com.fazecast.jSerialComm.*; public class ModbusRTUExample { public static void main(String[] args) { // 配置串口参数 SerialPort serialPort = SerialPort.getCommPort("COM1"); serialPort.setComPortParameters(9600, 8, 1, SerialPort.NO_PARITY); // 打开串口 if (serialPort.openPort()) { System.out.println("串口已打开!"); } else { System.out.println("无法打开串口!"); return; } // 创建Modbus RTU主站对象 ModbusMaster modbusMaster = new ModbusMaster(serialPort.getInputStream(), serialPort.getOutputStream()); // 读取保持寄存器的值 int slaveId = 1; // 从站地址 int startAddress = 0; // 起始寄存器地址 int numRegisters = 10; // 寄存器数量 try { int[] values = modbusMaster.readHoldingRegisters(slaveId, startAddress, numRegisters); for (int i = 0; i < values.length; i++) { System.out.println("寄存器 " + (startAddress + i) + " 的值为:" + values[i]); } } catch (ModbusException e) { e.printStackTrace(); } // 写入保持寄存器的值 int registerAddress = 3; // 寄存器地址 int value = 100; // 写入的值 try { modbusMaster.writeSingleRegister(slaveId, registerAddress, value); System.out.println("写入成功!"); } catch (ModbusException e) { e.printStackTrace(); } // 关闭串口 serialPort.closePort(); } } ``` 请注意,以上代码只是一个简单的示例,用于演示如何使用jSerialComm库实现Modbus RTU协议。具体的实现方式可能会根据你的需求和设备的要求而有所不同。你可以根据实际情况进行调整和扩展。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值