0

0

PHP 串口通信读取超时机制:解决阻塞问题与实现方法

霞舞

霞舞

发布时间:2025-08-22 16:32:01

|

427人浏览过

|

来源于php中文网

原创

PHP 串口通信读取超时机制:解决阻塞问题与实现方法

本教程详细探讨了在 PHP 中使用 lepiaf\\SerialPort 库进行串口通信时,read 方法可能导致的阻塞问题。通过分析库的内部实现,我们发现即使在非阻塞模式下,read 方法仍会无限等待分隔符。文章提供了一种修改库源代码以引入超时机制的解决方案,并指导如何在应用层优雅地处理串口读取超时,确保系统稳定性。

理解 PHP 串口通信中的阻塞问题

在使用 php 进行串口通信,特别是与外部硬件(如树莓派上的按钮输入)交互时,一个常见且关键的需求是等待特定数据的到来。然而,如果外部设备长时间没有响应,或者数据格式不符合预期,程序可能会无限期地等待串口数据,从而导致整个进程阻塞,甚至因为达到 php 的最大执行时间而终止。

原始代码尝试通过一个 while (true) 循环结合 time() 函数来手动实现超时机制:

// ... 串口初始化代码 ...

$timeoutStart = time();
$timeout = $timeoutStart + 15; // 15 秒超时
$aborted = false;

while (true) {
    $data2 = $this->serialPort->read(); // 关键的阻塞点

    if (Str::contains($data2, "Whatever I want to check for")) {
        // 处理数据并跳出循环
        break;
    }

    // 检查是否超时
    if (time() >= $timeout) {
        $this->serialPort->write("C,STOP\n"); // 发送停止命令
        $aborted = true;
        $this->alert("vending sequence stopped");
        break; // 超时跳出循环
    }
}

尽管代码中包含了超时检查逻辑,但实际运行中发现,即使超时时间已到,循环也可能“突然停止”或根本不执行超时逻辑。这通常是因为 $this->serialPort->read() 方法本身是一个阻塞调用,它会在没有数据到来或未检测到分隔符时无限期地等待,从而阻止了 time() 检查的执行。当 read() 方法被阻塞时,PHP 脚本会暂停,直到数据到达或达到 PHP 的 max_execution_time 限制而被系统强制终止。

深入分析 lepiaf\SerialPort 库的 read 方法

要解决这个问题,我们需要理解 lepiaf\\SerialPort 库的内部工作机制。通过查看其源代码,特别是 SerialPort.php 文件中的 read 方法,我们可以发现:

public function read()
{
    $this->ensureDeviceOpen();

    $chars = [];
    do {
        $char = fread($this->fd, 1); // 从文件描述符读取一个字符
        // ... 其他逻辑 ...
    } while ($char !== $this->getParser()->getSeparator()); // 循环直到找到分隔符

    return $this->getParser()->parse($chars);
}

尽管 lepiaf\\SerialPort 库可能将串口流设置为非阻塞模式,但其 read 方法内部的 do...while 循环却是一个阻塞点。它会不断地从串口读取单个字符,直到遇到配置的分隔符为止。这意味着,如果串口长时间没有数据,或者没有发送分隔符,这个 read 方法将永远不会返回,从而导致外部的超时逻辑无法执行。

立即学习PHP免费学习笔记(深入)”;

解决方案:修改 read 方法以引入超时机制

最直接有效的解决方案是修改 lepiaf\SerialPort 库的 read 方法,使其能够接受一个超时参数,并在指定时间内未接收到分隔符时返回。

火山写作
火山写作

字节跳动推出的中英文AI写作、语法纠错、智能润色工具,是一款集成创作、润色、纠错、改写、翻译等能力的中英文 AI 写作助手。

下载

请编辑 vendor/lepiaf/serialport/src/lepiaf/SerialPort/SerialPort.php 文件,找到 read 方法(通常在第 107 行左右),并将其修改为以下形式:

public function read($maxElapsed = 'infinite')
{
    $this->ensureDeviceOpen();

    $chars = [];
    // 计算超时时间点,使用 microtime(true) 获取微秒级时间戳
    $timeout = $maxElapsed == 'infinite' ? 1.7976931348623E+308 : (microtime(true) + $maxElapsed);

    do {
        $char = fread($this->fd, 1); // 尝试读取一个字符
        if ($char === '') { // 如果没有读取到字符(非阻塞模式下)
            if (microtime(true) > $timeout) {
                return false; // 超时,返回 false
            }
            usleep(100);    // 短暂休眠,避免 CPU 占用过高
            continue;       // 继续循环等待
        }
        $chars[] = $char; // 读取到字符,添加到缓冲区
    } while ($char !== $this->getParser()->getSeparator()); // 循环直到找到分隔符

    return $this->getParser()->parse($chars); // 解析并返回数据
}

修改说明:

  1. $maxElapsed 参数: 新增一个 $maxElapsed 参数,用于指定最大等待时间(秒)。默认值为 'infinite',表示无限等待。
  2. $timeout 计算: 根据 $maxElapsed 计算一个精确的超时时间点。microtime(true) 用于获取当前的微秒级时间戳,以实现更精确的超时控制。
  3. 超时判断: 在 fread($this->fd, 1) 返回空字符串(表示当前没有数据可读,因为流是非阻塞的)时,会立即检查当前时间是否已超过 $timeout。如果超时,则立即返回 false。
  4. usleep(100): 当没有数据可读且未超时时,程序会短暂休眠 100 微秒。这可以有效降低 CPU 占用,避免空转。
  5. 返回 false: 超时时返回 false,作为一种明确的超时信号。

在应用代码中实现带超时的读取

修改库文件后,您就可以在自己的应用代码中调用带有超时参数的 read 方法,并根据其返回值进行相应的处理:

// ... 串口初始化代码 ...

// 尝试读取串口数据,设置 15 秒的超时时间
$data2 = $this->serialPort->read(15); 

if ($data2 === false) {
    // 发生了超时,可以执行相应的错误处理或回滚操作
    $this->serialPort->write("C,STOP\n"); // 发送停止命令
    $aborted = true;
    $this->alert("vending sequence stopped due to timeout");
    // 这里可以进行 API 调用来回滚之前的操作
    // 例如:$this->apiClient->revertSomeAction();
} elseif (Str::contains($data2, "Whatever I want to check for")) {
    // 成功读取到数据,并且包含了期望的字符串
    // 处理数据并继续流程
    $this->alert("Received expected data: " . $data2);
} else {
    // 成功读取到数据,但未包含期望的字符串
    $this->alert("Received data, but not expected: " . $data2);
    // 可以选择继续等待或采取其他措施
}

通过这种方式,您的应用程序可以更健壮地处理串口通信,避免因外部设备无响应而导致的程序阻塞。

注意事项与总结

  1. 修改第三方库的风险: 直接修改 vendor 目录下的库文件存在风险。当您运行 composer update 时,这些修改可能会被覆盖。为了避免此问题,您可以考虑:
    • Fork 库: 将 lepiaf\\SerialPort 库 Fork 到自己的仓库,进行修改,然后在 composer.json 中引用您的 Fork 版本。
    • 扩展类: 如果库的设计允许,可以创建一个新的类来继承 lepiaf\\SerialPort\\SerialPort 并覆盖 read 方法。然而,对于这种底层的文件描述符操作,直接覆盖可能更复杂。
    • 补丁文件: 使用 cweagans/composer-patches 等工具来管理对 vendor 文件的补丁。 鉴于 lepiaf\\SerialPort 库的简单性,并且此处修改是核心功能增强,直接修改并记录下来是一个快速解决方案,但长期项目应考虑更健壮的维护策略。
  2. usleep() 的作用: usleep(100) 的目的是在没有数据可读时,让 CPU 短暂休眠,而不是进行忙等待,从而降低 CPU 占用。100 微秒是一个经验值,可以根据实际情况调整。
  3. 错误处理: 始终检查 read 方法的返回值。false 表示超时,其他值表示成功读取到的数据。
  4. PHP 的限制: 尽管引入了超时机制,PHP 仍然不是进行硬实时通信的最佳选择。对于对时间精度要求极高的应用,可能需要考虑使用 C/C++ 等语言。
  5. 分隔符的重要性: lepiaf\\SerialPort 库依赖于分隔符来判断一条消息的结束。确保您的硬件设备在发送数据时包含正确的分隔符,或者调整解析器配置。

通过上述修改和实践,您可以在 PHP 应用中实现一个可靠的串口读取超时机制,有效提升系统的稳定性和用户体验。

相关文章

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
composer是什么插件
composer是什么插件

Composer是一个PHP的依赖管理工具,它可以帮助开发者在PHP项目中管理和安装依赖的库文件。Composer通过一个中央化的存储库来管理所有的依赖库文件,这个存储库包含了各种可用的依赖库的信息和版本信息。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

151

2023.12.25

json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

418

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

535

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

311

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

77

2025.09.10

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

94

2023.09.25

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

298

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
第二十四期_PHP8编程
第二十四期_PHP8编程

共86课时 | 3.4万人学习

成为PHP架构师-自制PHP框架
成为PHP架构师-自制PHP框架

共28课时 | 2.5万人学习

第二十三期_PHP编程
第二十三期_PHP编程

共93课时 | 6.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号