PHP 无内置串口通信能力,需借助 php_serial 扩展或 system() 调用实现 RS-485 通信;发送十六进制数据须用 pack('H*', $hex) 转为字节流,并确保串口参数(波特率、数据位等)与设备一致,同时注意半双工方向控制、终端电阻及共模电压等物理层问题。

PHP 本身没有内置的串口通信能力,所谓“php485”不是标准 PHP 扩展或协议,而是指用 PHP 通过系统串口(如 /dev/ttyUSB0 或 COM3)与 RS-485 设备通信,常借助 php_serial 扩展或 system() 调用 stty/echo 等命令实现。发送十六进制数据的关键,在于「字节流构造」和「串口配置一致性」,而非 PHP 自身有什么 HEX 发送模式。
串口设备必须先正确打开并配置参数
RS-485 是物理层标准,不定义数据格式;PHP 只负责把字节写入串口设备文件。若波特率、数据位、停止位、校验位不匹配,接收端根本收不到有效数据,更别提解析 HEX。
-
stty -F /dev/ttyUSB0 9600 cs8 -cstopb -parenb(Linux 常用配置:9600 波特率、8 数据位、1 停止位、无校验) - Windows 下需确保
COM3已被占用且未被其他程序锁定 - PHP 中用
fopen('/dev/ttyUSB0', 'wb')打开后,必须用stream_set_timeout()和stream_set_blocking()控制读写行为,否则可能卡死 - 某些 USB-RS485 转换器需要额外控制 RTS 引脚(半双工方向切换),PHP 无法直接操作 GPIO,需依赖驱动自动翻转或外加硬件电路
十六进制字符串必须转换为二进制字节再写入
你写的 "010300000002C40B" 是十六进制字符串,不是原始字节。直接 fwrite($fp, "010300000002C40B") 会发 16 个 ASCII 字符('0','1','0','3',…),而非 8 个字节(\x01\x03\x00\x00\x00\x02\xC4\x0B)。
- 用
pack('H*', $hex_string)转换:例如pack('H*', '010300000002C40B')→ 8 字节原始数据 - 避免用
hex2bin():它要求输入长度为偶数且只含 0-9a-f,但对大小写敏感(PHP 7.4+ 支持小写,旧版可能失败) - 发送前建议用
bin2hex()回查确认:例如bin2hex(pack('H*', '0103')) === '0103' - 注意字节序——多数 Modbus RTU 协议使用大端,
pack()的H*模式天然按字符串顺序转字节,无需额外反转
写入后必须强制刷新并延时等待响应
RS-485 半双工通信中,发送完指令后需留出时间让从机处理并回传,同时避免 PHP 缓冲区延迟导致数据没真正发出。
- 调用
fflush($fp)确保内核缓冲区清空 - 用
usleep(10000)(10ms)或根据波特率估算最小帧间隔(如 9600 波特下 1 字节 ≈ 1.04ms,8 字节约 8.3ms) - 读取响应前,建议先
stream_set_timeout($fp, 1)防止无限阻塞 - 读取时用
fread($fp, 256),再用bin2hex()查看是否收到预期 HEX 响应(如"01030400010002fa42")
#!/usr/bin/env php
真正容易被忽略的不是怎么发 HEX,而是:RS-485 方向控制是否可靠、共模电压是否在 -7V~+12V 范围内、终端电阻是否匹配(120Ω)、以及从机地址/功能码是否与实际设备一致。这些物理和协议层问题,PHP 层面完全无法检测,只能靠示波器或 Modbus 调试工具交叉验证。











