laravel
5. 深入探讨
并发

简介

[!警告] 在我们收集社区反馈期间,Laravel 的 Concurrency 外观目前处于测试阶段。

有时您可能需要执行几个彼此不依赖的缓慢任务。在许多情况下,通过并发执行这些任务可以实现显著的性能提升。Laravel 的 Concurrency 外观为并发执行闭包提供了一个简单、方便的 API。

工作原理

Laravel 通过将给定的闭包进行序列化,并将其分发到一个隐藏的 Artisan CLI 命令来实现并发。该命令会将闭包进行反序列化,并在其自己的 PHP 进程中调用它。在闭包被调用后,结果值会被序列化回父进程。

Concurrency 外观支持三种驱动程序:process(默认)、forksync

与默认的 process 驱动程序相比,fork 驱动程序性能更佳,但它只能在 PHP 的 CLI 上下文中使用,因为 PHP 在 Web 请求期间不支持分叉。在使用 fork 驱动程序之前,您需要安装 spatie/fork 包:

composer require spatie/fork

sync 驱动程序主要在测试时有用,当您想要禁用所有并发并在父进程中按顺序简单地执行给定的闭包时。

运行并发任务

要运行并发任务,您可以调用 Concurrency 外观的 run 方法。run 方法接受一个闭包数组,这些闭包应在子 PHP 进程中同时执行:

use Illuminate\Support\Facades\Concurrency;
use Illuminate\Support\Facades\DB;
 
[$userCount, $orderCount] = Concurrency::run([
    fn () => DB::table('users')->count(),
    fn () => DB::table('orders')->count(),
]);

要使用特定的驱动程序,您可以使用 driver 方法:

$results = Concurrency::driver('fork')->run(...);

或者,要更改默认的并发驱动程序,您应该通过 config:publish Artisan 命令发布 concurrency 配置文件,并更新文件中的 default 选项:

php artisan config:publish concurrency

延迟并发任务

如果您想要并发地执行一组闭包,但对这些闭包返回的结果不感兴趣,您应该考虑使用 defer 方法。当调用 defer 方法时,给定的闭包不会立即执行。相反,Laravel 会在将 HTTP 响应发送给用户后并发地执行这些闭包:

use App\Services\Metrics;
use Illuminate\Support\Facades\Concurrency;
 
Concurrency::defer([
    fn () => Metrics::report('users'),
    fn () => Metrics::report('orders'),
]);