PHP无法直接读取红外接收头GPIO信号,必须通过lirc+irw或Python+RPi.GPIO等底层程序解码后,再由PHP调用命令行或读取共享文件/Redis获取结果。

PHP 本身无法直接读取红外接收头的 GPIO 信号
PHP 是服务端脚本语言,没有内置能力访问树莓派或嵌入式设备的物理引脚(如 GPIO18)。所谓“PHP 读红外”,本质是让底层 C/Python 程序完成硬件交互,再把解码结果通过文件、Socket 或 HTTP 接口暴露给 PHP。直接在 php-fpm 或 Apache 进程里调用 gpio read 或 poll() 会失败——权限、实时性、中断响应都不支持。
推荐方案:用 lirc + irw 做解码,PHP 调用命令行读取
这是最稳定、兼容性最好的路径。前提是你的红外接收头已接在树莓派 GPIO(通常为 GPIO18),且 lirc 已正确配置驱动(lirc_rpi 或 gpio-ir-recv)和遥控配置文件(/etc/lirc/lircd.conf.d/*.conf)。
- 确认
lirc正常运行:sudo systemctl status lirc,应显示 active - 测试是否收到按键:
irw,按遥控器,终端应输出类似0000000000ff6897 my_remote KEY_POWER - PHP 中用
shell_exec()或proc_open()启动irw并实时读取 stdout(注意:不能用exec(),它会阻塞等待结束)
#!/usr/bin/env php
['pipe', 'w'],
2 => ['pipe', 'w']
], $pipes);
if (is_resource($handle)) {
stream_set_blocking($pipes[1], false); // 非阻塞读
while (true) {
$line = fgets($pipes[1]);
if ($line !== false && trim($line)) {
preg_match('/\s+(\w+)\s+(\w+)$/', $line, $m);
if (isset($m[1], $m[2])) {
echo "Button: {$m[1]}, Code: {$m[2]}\n";
// 可写入 Redis / 文件 / 触发 API
}
}
usleep(10000); // 10ms 间隔防 CPU 占满
}
proc_close($handle);
}
?>
绕过 lirc:用 Python + RPi.GPIO 捕获原始脉冲,PHP 读取共享文件
适合需要自定义协议(比如非 NEC 的私有红外编码)、或想避开 lirc 配置复杂性的场景。但要求你懂红外载波、脉冲宽度判别逻辑,且 Python 进程需以 root 权限运行。
- Python 脚本用
GPIO.wait_for_edge()捕获上升/下降沿,计算高/低电平持续时间(单位微秒) - 将解码后的键值(如
"VOL_UP")写入一个临时文件(如/tmp/ir_last.key),或通过redis-cli set ir:last_key VOL_UP - PHP 定期用
file_get_contents('/tmp/ir_last.key')或redis->get('ir:last_key')获取,注意加锁或时间戳防重复消费
关键点:RPi.GPIO 在树莓派上对 50μs 级脉冲精度有限,NEC 协议通常可稳定识别;但 Sony、RC5 等短周期协议容易丢边沿。
立即学习“PHP免费学习笔记(深入)”;
常见失败原因和必须检查项
90% 的问题出在硬件层或权限控制,而非 PHP 代码本身:
- 红外接收头供电不对:必须接 3.3V(不是 5V),GND 和 OUT 接线无反接;OUT 接 GPIO 引脚前建议串 1kΩ 电阻防灌入电流
-
lirc驱动未加载:dmesg | grep lirc应看到lirc_rpi: driver registered;若报no such device,检查/boot/config.txt是否有dtoverlay=lirc-rpi,gpio_in_pin=18 - PHP 进程无权限执行
irw:确保运行 PHP 的用户(如www-data)在lirc用户组中:sudo usermod -a -G lirc www-data,并重启php-fpm -
irw输出为空但mode2 -d /dev/lirc0能看到原始脉冲:说明解码配置缺失,检查/etc/lirc/lircd.conf.d/下是否有对应遥控器的 .conf 文件
红外信号易受环境光干扰,调试时关掉日光灯、拉上窗帘;同一空间多个红外设备会互相串扰,这是物理限制,PHP 层没法解决。











