跳至内容

并发

介绍

exclamation

Laravel 的 Concurrency 门面目前处于测试阶段,我们正在收集社区反馈。

有时你可能需要执行几个不相互依赖的缓慢任务。在许多情况下,通过并发执行任务可以实现显著的性能改进。Laravel 的 Concurrency 门面提供了一个简单方便的 API,用于并发执行闭包。

并发兼容性

如果你从 Laravel 10.x 应用程序升级到 Laravel 11.x,你可能需要在应用程序的 config/app.php 配置文件中将 ConcurrencyServiceProvider 添加到 providers 数组中。

'providers' => ServiceProvider::defaultProviders()->merge([
/*
* Package Service Providers...
*/
Illuminate\Concurrency\ConcurrencyServiceProvider::class,
 
/*
* Application Service Providers...
*/
App\Providers\AppServiceProvider::class,
App\Providers\AuthServiceProvider::class,
// App\Providers\BroadcastServiceProvider::class,
App\Providers\EventServiceProvider::class,
App\Providers\RouteServiceProvider::class,
])->toArray(),

工作原理

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

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

fork 驱动程序与默认的 process 驱动程序相比,性能有所提升,但它只能在 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'),
]);