PHP无法直接可靠实现RS-485多设备轮询,因其短生命周期、无原生非阻塞I/O及线程级串口独占支持;可行方案是PHP仅作调度,由Python/C等长期进程处理串口轮询并存结果至Redis,PHP读取。

PHP 本身不直接支持串口通信,更不存在所谓“PHP485”这个官方扩展或标准库。所谓“php485多设备轮询”或“多串口并发控制”,实际是用 PHP 调用底层串口(如 /dev/ttyUSB0、COM3)与 RS-485 总线上的多个设备通信,本质是「PHP + 串口驱动 + 协议轮询逻辑」的组合方案。直接用 PHP 做实时轮询极易出错,必须清楚边界和替代路径。
为什么不能直接用 PHP 做 RS-485 多设备轮询
PHP 是为 Web 请求设计的短生命周期脚本语言,不具备稳定串口持有能力、无原生非阻塞 I/O、不支持线程级串口独占——这意味着:
-
fopen('COM3', 'r+')在 CLI 模式下可能成功,但 Web SAPI(如 Apache/FPM)中会因权限、进程隔离、超时被中断 - RS-485 半双工需严格控制 DE/RE 引脚(常靠 USB-485 转换器自动处理),但 PHP 无法干预硬件流控,易发冲突导致从机无响应
- 轮询多个地址(如 Modbus RTU 的
0x01~0x10)时,若某设备掉线或响应超时,fread()会卡死,除非手动设stream_set_timeout()且重试逻辑极难健壮
可行方案:PHP 只做调度,轮询交给专用进程
真正落地的做法是把串口操作下沉到长期运行的守护进程,PHP 仅通过轻量接口(如文件、Redis、HTTP)触发或获取结果。例如:
- 用 Python(
pyserial)或 C 写一个轮询服务,监听 Redis 队列:redis-cli publish modbus:cmd '{"addr":1,"reg":40001,"len":1}' - 该服务按优先级/周期轮询各设备,将结果存入 Redis Hash:
HSET modbus:data:1 temperature 23.5 - PHP 直接读
$data = $redis->hGet('modbus:data:1', 'temperature');,完全规避串口操作
如果坚持用 PHP 直连串口(仅限 Linux CLI 场景)
必须满足三个前提:使用 php_serial 扩展(非 PECL 官方,需手动编译)、关闭输出缓冲、所有调用在单进程内完成。关键配置示例:
立即学习“PHP免费学习笔记(深入)”;
ini_set('max_execution_time', 0);
$serial = new phpSerial();
$serial->deviceSet("/dev/ttyUSB0");
$serial->confBaudRate(9600);
$serial->confParity("none");
$serial->confCharacterLength(8);
$serial->confStopBits(1);
$serial->confFlowControl("none");
$serial->deviceOpen();
// 发送 Modbus RTU 请求帧(含 CRC)
$serial->sendMessage("\x01\x03\x9C\x40\x00\x01\x7A\xC6"); // addr=1, read reg=40001
usleep(100000); // 等待响应,不能依赖 fread 阻塞
$response = $serial->readPort(); // 需自行解析帧头+CRC
$serial->deviceClose();
注意:usleep() 时间必须大于从机最大响应时间(查手册,常见 50–200ms),否则读不到数据;readPort() 返回的是原始字节流,PHP 无内置 CRC-16 计算函数,得自己实现或引入第三方类。
多串口并发的本质是进程隔离,不是 PHP 并发
想同时控制 /dev/ttyUSB0(485-A 组)和 /dev/ttyUSB1(485-B 组),正确做法是起两个独立进程(而非 fork 或 pthreads):
- 用
nohup php poller.php --port=/dev/ttyUSB0 --group=A &启一个 - 再启一个
nohup php poller.php --port=/dev/ttyUSB1 --group=B & - 两进程各自维护自己的串口句柄、超时计时器、重试次数,互不干扰
试图在单个 PHP 进程里用 stream_select() 等待多个串口——在 Linux 下基本无效,因为串口设备文件不支持 select/poll 的就绪通知(仅对 socket、pipe 有效)。
RS-485 的可靠性不取决于上层语言,而取决于物理层匹配、终端电阻、共模电压、轮询间隔是否大于最慢设备的处理周期。PHP 只能当个“传话员”,别让它扛收发大梁。











