PHP 无法直接支持 RS-485 通信,需通过串口与 USB-RS485 模块交互并实现 Modbus RTU 协议;成功读取温湿度数据的关键在于串口配置、Modbus 帧构造和数据解析三者严格对齐。
PHP 本身不直接支持 RS-485 硬件通信,所谓“PHP485”不是语言特性,而是指用 PHP 通过串口(如 /dev/ttyUSB0)与 RS-485 转 USB 模块通信,再对接 Modbus RTU 协议的温湿度传感器。能否读通,关键不在 PHP 多厉害,而在三件事是否对齐:串口配置、Modbus 帧构造、数据解析逻辑。
Linux 下常见错误是 Permission denied 或 No such file or directory,本质是没找到设备或没读写权。
dmesg | grep tty 查看插上 USB-RS485 模块后系统分配的设备名(通常是 /dev/ttyUSB0 或 /dev/ttyACM0)ls -l /dev/ttyUSB0 —— 若属组不是 dialout,执行 sudo usermod -a -G dialout $USER
并重登posix 和 sockets 扩展,但真正读串口靠的是 fopen() + fwrite()/fread(),不是 socketfile_get_contents()——它不支持非阻塞/超时控制,极易卡死多数温湿度传感器(如 RSDS12、SHT30+RS485 模块)走标准 Modbus RTU,功能码 0x03(读保持寄存器),起始地址和长度依手册而定。PHP 没有原生 Modbus 库,手动拼帧最可控。
0x01,起始地址 0x0000,读 2 字节):"\x01\x03\x00\x00\x00\x01\x84\x0A"
modbus_crc16())比自己手写安全;网上很多 PHP CRC 实现有 bug,建议直接抄工业项目验证过的版本usleep(5000) 避免连续请求冲垮从机(尤其 9600 波特率下)usleep(10000) 再读,否则读到空或乱码收到的响应帧如 "\x01\x03\x02\x01\x01\x01\xD4",不代表温度是 257℃。Modbus 返回的是原始寄存器值,需按设备手册换算。
substr($response, 3, 2) 得到 "\x01\x01"
unpack('n', ...) 解包为大端无符号整数 → 得到 257
25.7℃
'v'(小端)解包;忽略单位换算(比如湿度返回值是 0–1000,要除以 10 才是百分比)RS485 是半双工,同一时刻只能收或发;Modbus RTU 对时序极其敏感。PHP 不是实时语言,稍不注意就丢帧。
stty -F /dev/ttyUSB0 -crtscts,否则 RTS/CTS 信号可能锁死通信stty -F /dev/ttyUSB0 9600 cs8 -cstopb -parenb -icanon -echo
fopen(..., 'r+b') 打开,stream_set_timeout() 设 read timeout(建议 1.5 秒),超时后必须 fclose() 重开,不能复用句柄fread() 读完一帧——Modbus 响应长度可变,要用循环 + 缓冲区累积,直到收到完整帧(含正确 CRC)真正卡住的地方,往往不是 PHP 写错了,而是你没看到那根 RS485 的 A/B 线接反了,或者终端电阻没加导致信号反射——这些物理层问题,会让所有软件逻辑失效。