主题
并发处理
引言
WARNING
Laravel 的 Concurrency
门面目前处于 Beta 阶段,我们正在收集社区反馈。
有时,您可能需要执行多个相互独立的慢任务。在许多情况下,通过并发执行这些任务可以显著提升性能。Laravel 的 Concurrency
门面提供了一个简单且方便的 API,用于并发执行闭包。
并发兼容性
如果您从 Laravel 10.x 升级到 Laravel 11.x,可能需要在应用程序的 config/app.php
配置文件中的 providers
数组中添加 ConcurrencyServiceProvider
:
php
'providers' => ServiceProvider::defaultProviders()->merge([
/*
* 包服务提供者...
*/
Illuminate\Concurrency\ConcurrencyServiceProvider::class,
/*
* 应用服务提供者...
*/
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
(默认)、fork
和 sync
。
fork
驱动相比默认的 process
驱动具有更高的性能,但它只能在 PHP 的 CLI 上下文中使用,因为 PHP 在 Web 请求中不支持进程分叉。在使用 fork
驱动之前,您需要安装 spatie/fork
包:
shell
composer require spatie/fork
sync
驱动主要在测试时有用,此时您可能希望禁用所有并发功能,并简单地在父进程中按顺序执行给定的闭包。
运行并发任务
要运行并发任务,可以调用 Concurrency
门面的 run
方法。run
方法接受一个闭包数组,这些闭包将在子 PHP 进程中同时执行:
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
方法:
php
$results = Concurrency::driver('fork')->run(...);
或者,要更改默认的并发驱动,您应该通过 config:publish
Artisan 命令发布 concurrency
配置文件,并更新文件中的 default
选项:
shell
php artisan config:publish concurrency
延迟并发任务
如果您希望并发执行一组闭包,但对这些闭包的返回结果不感兴趣,可以考虑使用 defer
方法。调用 defer
方法时,给定的闭包不会立即执行。相反,Laravel 会在 HTTP 响应发送给用户后并发执行这些闭包:
php
use App\Services\Metrics;
use Illuminate\Support\Facades\Concurrency;
Concurrency::defer([
fn () => Metrics::report('users'),
fn () => Metrics::report('orders'),
]);