
本文旨在解决在使用PHP的`proc_open`函数与`cu`命令交互时,`fread`函数出现挂起的问题。通过对比成功读取`cu -v`命令输出和无法读取`cu -s 115200 -l /dev/ttyUSB3`命令输出的情况,提供了一种基于Symfony Process组件的解决方案,避免了`fread`的阻塞,实现了与GSM调制解调器的稳定通信。
在使用PHP的proc_open函数与外部程序交互时,有时会遇到一些意想不到的问题。一个典型的例子是在尝试通过cu命令连接GSM调制解调器时,使用fread函数读取cu命令的输出流时发生挂起。本文将探讨这个问题,并提供一种解决方案。
问题描述
在使用proc_open函数执行cu -s 115200 -l /dev/ttyUSB3命令时,期望能够读取到cu命令的输出,例如"Connected."。然而,实际情况是fread函数会一直阻塞,无法读取到任何数据。
以下是问题代码示例:
立即学习“PHP免费学习笔记(深入)”;
相反,如果执行cu -v命令,fread函数可以正常读取到输出,例如"c"。
这表明proc_open和fread本身并没有问题,问题可能出在与特定命令的交互上。
解决方案:使用Symfony Process组件
经过尝试,发现Symfony Process组件可以有效地解决这个问题。Symfony Process组件封装了与外部进程交互的复杂性,提供了一个更稳定和可靠的接口。
1. 安装Symfony Process组件
首先,需要使用Composer安装Symfony Process组件:
composer require symfony/process
2. 使用Symfony Process组件执行命令
然后,可以使用以下代码来执行cu -s 115200 -l /dev/ttyUSB3命令并读取输出:
run();
if (!$process->isSuccessful()) {
throw new \Exception('cu command failed: ' . $process->getErrorOutput());
}
echo $process->getOutput();
?>代码解释:
- require 'vendor/autoload.php';:引入Composer自动加载器,以便使用Symfony Process组件。
- use Symfony\Component\Process\Process;:引入Process类。
- $process = new Process(['cu', '-s', '115200', '-l', '/dev/ttyUSB3']);:创建一个Process对象,指定要执行的命令及其参数。注意,命令和参数应该分别作为数组元素传递。
- $process->run();:执行命令。
- if (!$process->isSuccessful()) { ... }:检查命令是否成功执行。如果失败,抛出异常并输出错误信息。
- echo $process->getOutput();:输出命令的执行结果。
总结
虽然Symfony Process组件底层可能也使用了proc_open函数,但它可能对进程的管理和流的处理方式进行了优化,从而避免了fread挂起的问题。如果你在使用PHP的proc_open函数与外部程序交互时遇到类似的问题,可以尝试使用Symfony Process组件来解决。
注意事项:
- 确保已经正确安装了cu命令,并且当前用户具有执行该命令的权限。
- 根据实际情况修改命令参数,例如串口设备/dev/ttyUSB3。
- 错误处理是至关重要的,应该始终检查命令是否成功执行,并处理可能的错误。
通过使用Symfony Process组件,可以更可靠地与外部进程进行交互,避免fread挂起等问题,从而提高程序的稳定性和可靠性。











