当 php 8.1 引入 fibers 时,许多开发人员想知道它们是否可以解决 php 作为单线程同步语言的长期限制。 fibers 能否使 php 变得异步,就像带有事件循环的 JavaScript 或 node.JS 一样?答案很微妙:纤程不提供真正的异步执行,但它们是更有效地管理任务的强大工具。让我们详细探讨这个概念。
什么是 php 纤维?
fibers 是 php 中实现协作多任务的机制。它们允许您暂停和恢复代码的特定部分,而不会阻塞整个 php 进程。将 fiber 视为一种特殊功能,可以将控制权“让渡”回主程序,然后根据请求从中断处继续执行。
纤维的主要特性:
光纤暂停时会发生什么?
当使用 fiber::suspend() 暂停 fiber 时,控制返回到主 php 脚本。这意味着:
- 主进程可以自由地继续执行程序的其他部分。
- fiber 的执行暂时停止,等待resume() 调用。
例如:
$fiber = new fiber(function () { echo "fiber started "; fiber::suspend(); echo "fiber resumed "; }); echo "before fiber "; $fiber->start(); echo "after fiber start "; $fiber->resume(); echo "after fiber resume "; output:
before fiber fiber started after fiber start fiber resumed after fiber resume
这是发生的事情:
立即学习“PHP免费学习笔记(深入)”;
- fiber::suspend() 暂停 fiber。 $ fiber->start() 之后执行转移回主脚本。
- 主脚本继续运行(“光纤启动后”)。
- 当resume()被调用时,fiber会从上次中断的地方继续执行它的任务。
恢复光纤会阻塞主进程吗?
是的,但只是暂时的。当您调用 fiber::resume() 时,fiber 在主 php 进程中同步运行。在此期间:
- 脚本的其他部分(或其他 fiber)在 fiber 完成或再次暂停之前无法执行。
- fiber 的执行是阻塞的,因为 php 仍然是单线程的。 例如:
$fiber = new fiber(function () { echo "processing fiber... "; sleep(2); // simulates a blocking task echo "fiber done "; }); echo "before fiber "; $fiber->start(); echo "between fiber start and resume "; $fiber->resume(); echo "after fiber "; output:
Before Fiber Processing Fiber... Fiber Done Between Fiber Start and Resume After Fiber
这里,fiber 在 sleep(2) 调用期间阻塞了主进程。因此,虽然 fiber 提供了一种构建代码以提高效率的方法,但它们并不能神奇地实现并行或真正的异步执行。
光纤如何仍然“无阻塞”?
术语“非阻塞”是指 fiber 如何实现更好的任务管理,而不是并行执行。 fiber 在暂停时不会阻塞主进程;相反,控制权被交还给主脚本或事件循环。
这对于使用事件驱动架构的库或框架特别有用,例如 reactphp 或 amp,其中:
- 可以暂停长时间运行或等待的任务(例如数据库查询、api 调用)。
- 其他任务可以同时继续运行。
- 一旦任务准备就绪,fiber 就会恢复,并继续执行。
想象你是一位厨师,正在准备多种菜肴:
您开始烹饪菜肴,但停下来等待食物沸腾。
在等待的同时,你开始准备另一道菜。
当第一道菜做好后,您可以返回并继续烹饪。
类似地,fibers 允许 php “暂停”任务并稍后返回该任务,而不会中断整个过程。
为什么光纤不是真正的异步。
与 javascript 或 node.js 中的异步编程不同,任务可以使用线程或事件循环并行运行,fibers:
- 在单个 php 进程中同步执行。
- 通过允许开发人员手动控制任务暂停和恢复来提供协作式多任务处理。
换句话说:
纤维不会引入并行性(任务仍然一次运行一个)。
它们是一种更有效地管理和构建非阻塞代码的工具。
虽然 php fibers 并未使 php 真正实现异步,但它们是该语言的强大补充。