简介
在某些应用中,一些查询数据或处理任务的操作会在某段时间里短时间内大量进行,或是一个操作花费好几秒钟。当出现这种情况时,通常会将检索到的数据缓存起来,从而为后面请求同一数据的请求迅速返回结果。这些缓存数据通常会储存在极快的存储系统中,例如 Memcached 和 Redis。
Laravel 为各种缓存后端提供了富有表现力且统一的 API,以便你利用它们极快的查询数据来加快你的应用。
配置
缓存配置文件位于 php config/cache.php
。在这个文件中,你可以指定应用默认使用哪个缓存驱动。Laravel 支持的缓存后端包括 Memcached、 Redis、 DynamoDB,以及现成的关系型数据库。此外,还支持基于文件的缓存驱动,以及方便自动化测试的缓存驱动 php array
和 php null
。
缓存配置文件还包含文件中记录的各种其他选项,因此请务必阅读这些选项。 默认情况下,Laravel 配置为使用 php file
缓存驱动,它将序列化的缓存对象存储在服务器的文件系统中。 对于较大的应用程序,建议你使用更强大的驱动,例如 Memcached 或 Redis。 你甚至可以为同一个驱动配置多个缓存配置。
驱动先决条件
Database
使用 php database
缓存驱动时,你需要设置一个表来包含缓存项。你将在下表中找到 php Schema
声明的示例:
Schema::create('cache', function (Blueprint $table) {
$table->string('key')->unique();
$table->text('value');
$table->integer('expiration');
});
注意
你还可以使用 php php artisan cache:table
Artisan 命令生成具有适当模式的迁移。
Memcached
使用 Memcached 驱动程序需要安装 Memcached PECL 包。你可以在 php config/cache.php
配置文件中列出所有的 Memcached 服务器。该文件已经包含一个 php memcached.servers
来帮助你入门:
'memcached' => [
'servers' => [
[
'host' => env('MEMCACHED_HOST', '127.0.0.1'),
'port' => env('MEMCACHED_PORT', 11211),
'weight' => 100,
],
],
],
如果需要,你可以将 php host
选项设置为 UNIX socket path。 如果这样做, php port
选项应设置为 php 0
:
'memcached' => [
[
'host' => '/var/run/memcached/memcached.sock',
'port' => 0,
'weight' => 100
],
],
Redis
在将 Redis 缓存与 Laravel 一起使用之前,您需要通过 PECL 安装 PhpRedis PHP 扩展或通过 Composer 安装 php predis/predis
包(~1.0)。Laravel Sail 已经包含了这个扩展。另外,Laravel 官方部署平台如 Laravel Forge 和 Laravel Vapor 也默认安装了 PhpRedis 扩展。
有关配置 Redis 的更多信息,请参阅其 Laravel documentation page.
DynamoDB
在使用 DynamoDB 缓存驱动程序之前,您必须创建一个 DynamoDB 表来存储所有缓存的数据。通常,此表应命名为php cache
。但是,您应该根据应用程序的缓存配置文件中的 php stores.dynamodb.table
配置值来命名表。
该表还应该有一个字符串分区键,其名称对应于应用程序的缓存配置文件中的 php stores.dynamodb.attributes.key
配置项的值。 默认情况下,分区键应命名为 php key
。
缓存用法
获取缓存实例
要获取缓存存储实例,您可以使用 php Cache
门面类,我们将在本文档中使用它。php Cache
门面类提供了对 Laravel 缓存底层实现的方便、简单的访问:
<?php
namespace App\Http\Controllers;
use Illuminate\Support\Facades\Cache;
class UserController extends Controller
{
/**
* Show a list of all users of the application.
*/
public function index(): array
{
$value = Cache::get('key');
return [
// ...
];
}
}
访问多个缓存存储
使用 php Cache
门面类, 您可以通过 php store
方法访问各种缓存存储。传递给 php store
方法的键应该对应于 php cache
配置文件中的 php stores
配置数组中列出的存储之一:
$value = Cache::store('file')->get('foo');
Cache::store('redis')->put('bar', 'baz', 600); // 10 Minutes
从缓存中检索项目
php Cache
门面的 php get
方法用于从缓存中检索项目。如果缓存中不存在该项目,则将返回 php null
。如果您愿意,您可以将第二个参数传递给 php get
方法,指定您希望在项目不存在时返回的默认值:
$value = Cache::get('key');
$value = Cache::get('key', 'default');
您甚至可以将闭包作为默认值传递。如果指定的项在缓存中不存在,则返回闭包的结果。传递闭包允许您推迟从数据库或其他外部服务中检索默认值:
$value = Cache::get('key', function () {
return DB::table(/* ... */)->get();
});
检查项目是否存在php has
方法可用于确定缓存中是否存在项目。如果项目存在但其值为 php null
,此方法也将返回 php false
:
if (Cache::has('key')) {
// ...
}
递增 / 递减值php increment
和 php decrement
方法可用于调整缓存中整数项的值。这两种方法都接受一个可选的第二个参数,指示增加或减少项目值的数量:
Cache::increment('key');
Cache::increment('key', $amount);
Cache::decrement('key');
Cache::decrement('key', $amount);
检索和存储
有时你可能希望从缓存中检索一个项目,但如果请求的项目不存在,则存储一个默认值。 例如, 你可能希望从缓存中检索所有用户,如果用户不存在,则从数据库中检索并将它们添加到缓存中。 你可以使用 php Cache::remember
方法执行此操作:
$value = Cache::remember('users', $seconds, function () {
return DB::table('users')->get();
});
如果该项不存在于缓存中,将执行传递给 php remember
方法的闭包,并将其结果放入缓存中。
你可以使用 php rememberForever
方法从缓存中检索一个项目,如果它不存在则永久存储它:
$value = Cache::rememberForever('users', function () {
return DB::table('users')->get();
});
检索和删除
如果你需要从缓存中检索一项后并删除该项,你可以使用 php pull
方法。 与 php get
方法一样,如果该项不存在于缓存中,将返回 php null
:
$value = Cache::pull('key');
在缓存中存储项目
你可以使用 php Cache
Facade上的 php put
方法将项目存储在缓存中:
Cache::put('key', 'value', $seconds = 10);
如果存储时间没有传递给 php put
方法,则该项目将无限期存储:
Cache::put('key', 'value');
除了将秒数作为整数传递之外,你还可以传递一个代表缓存项所需过期时间的 php DateTime
实例:
Cache::put('key', 'value', now()->addMinutes(10));
如果不存在则存储php add
方法只会将缓存存储中不存在的项目添加到缓存中。如果项目实际添加到缓存中,该方法将返回 php true
。 否则,该方法将返回 php false
。 php add
方法是一个原子操作:
Cache::add('key', 'value', $seconds);
永久存储php forever
方法可用于将项目永久存储在缓存中。由于这些项目不会过期,因此必须使用 php forget
方法手动将它们从缓存中删除:
Cache::forever('key', 'value');
注意:如果您使用的是 Memcached 驱动程序,则当缓存达到其大小限制时,可能会删除「永久」存储的项目。
从缓存中删除项目
您可以使用 php forget
方法从缓存中删除项目:
Cache::forget('key');
您还可以通过提供零或负数的过期秒数来删除项目:
Cache::put('key', 'value', 0);
Cache::put('key', 'value', -5);
您可以使用 php flush
方法清除整个缓存:
Cache::flush();
注意:刷新缓存不会考虑您配置的缓存「前缀」,并且会从缓存中删除所有条目。在清除由其他应用程序共享的缓存时,请考虑到这一点。
缓存助手函数
除了使用 php Cache
门面之外,您还可以使用全局 php cache
函数通过缓存检索和存储数据。当使用单个字符串参数调用 php cache
函数时,它将返回给定键的值:
$value = cache('key');
如果您向函数提供键 / 值对数组和过期时间,它将在指定的持续时间内将值存储在缓存中:
cache(['key' => 'value'], $seconds);
cache(['key' => 'value'], now()->addMinutes(10));
当不带任何参数调用 cache 函数时,它会返回 Illuminate\Contracts\Cache\Factory 实现的实例,允许您调用其他缓存方法:
cache()->remember('users', $seconds, function () {
return DB::table('users')->get();
});
技巧
在测试对全局 php cache
函数的调用时,您可以使用 php Cache::shouldReceive
方法,就像 testing the facade.
缓存标签
注意
使用 php file
, php dynamodb
或 php database
存驱动程序时不支持缓存标记。 此外,当使用带有 “永久” 存储的缓存的多个标签时,使用诸如 “memcached” 之类的驱动程序会获得最佳性能,它会自动清除陈旧的记录。
存储缓存标签
缓存标签允许您在缓存中标记相关项目,然后刷新所有已分配给定标签的缓存值。您可以通过传入标记名称的有序数组来访问标记缓存。例如,让我们访问一个标记的缓存并将一个值php put
缓存中:
Cache::tags(['people', 'artists'])->put('John', $john, $seconds);
Cache::tags(['people', 'authors'])->put('Anne', $anne, $seconds);
访问缓存标签
要检索标记的缓存项,请将相同的有序标签列表传递给 tags 方法,然后使用您要检索的键调用 php get
方法:
$john = Cache::tags(['people', 'artists'])->get('John');
$anne = Cache::tags(['people', 'authors'])->get('Anne');
删除被标记的缓存数据
你可以刷新所有分配了标签或标签列表的项目。 例如,此语句将删除所有标记有 php people
, php authors
或两者的缓存。因此,php Anne
和 php John
都将从缓存中删除:
Cache::tags(['people', 'authors'])->flush();
相反,此语句将仅删除带有 php authors
标记的缓存,因此将删除 php Anne
,但不会删除 php John
:
Cache::tags('authors')->flush();
清理过期的缓存标记
注意:仅在使用 Redis 作为应用程序的缓存驱动程序时,才需要清理过期的缓存标记。
为了在使用 Redis 缓存驱动程序时正确清理过期的缓存标记,Laravel 的 Artisan 命令 php cache:prune-stale-tags
应该被添加到 任务调度 中,在应用程序的 php App\Console\Kernel
类里:
$schedule->command('cache:prune-stale-tags')->hourly();
原子锁
注意:要使用此功能,您的应用程序必须使用php memcached
、php redis
、php dynamicodb
、php database
、php file
或php array
缓存驱动程序作为应用程序的默认缓存驱动程序。
此外,所有服务器都必须与同一中央缓存服务器通信。
驱动程序先决条件
数据库
使用“数据库”缓存驱动程序时,您需要设置一个表来包含应用程序的缓存锁。您将在下表中找到一个示例 php Schema
声明:
Schema::create('cache_locks', function (Blueprint $table) {
$table->string('key')->primary();
$table->string('owner');
$table->integer('expiration');
});
管理锁
原子锁允许操作分布式锁而不用担心竞争条件。例如, Laravel Forge 使用原子锁来确保服务器上一次只执行一个远程任务。您可以使用 php Cache::lock
方法创建和管理锁:
use Illuminate\Support\Facades\Cache;
$lock = Cache::lock('foo', 10);
if ($lock->get()) {
// Lock acquired for 10 seconds...
$lock->release();
}
php get
方法也接受一个闭包。闭包执行后,Laravel 会自动释放锁:
Cache::lock('foo', 10)->get(function () {
// Lock acquired for 10 seconds and automatically released...
});
如果在您请求时锁不可用,您可以指示 Laravel 等待指定的秒数。如果在指定的时间限制内无法获取锁,则会抛出 Illuminate\Contracts\Cache\LockTimeoutException:
use Illuminate\Contracts\Cache\LockTimeoutException;
$lock = Cache::lock('foo', 10);
try {
$lock->block(5);
// Lock acquired after waiting a maximum of 5 seconds...
} catch (LockTimeoutException $e) {
// Unable to acquire lock...
} finally {
$lock?->release();
}
上面的例子可以通过将闭包传递给 php block
方法来简化。当一个闭包被传递给这个方法时,Laravel 将尝试在指定的秒数内获取锁,并在闭包执行后自动释放锁:
Cache::lock('foo', 10)->block(5, function () {
// Lock acquired after waiting a maximum of 5 seconds...
});
跨进程管理锁
有时,您可能希望在一个进程中获取锁并在另一个进程中释放它。例如,您可能在 Web 请求期间获取锁,并希望在由该请求触发的排队作业结束时释放锁。在这种情况下,您应该将锁的作用域php owner token
传递给排队的作业,以便作业可以使用给定的令牌重新实例化锁。
在下面的示例中,如果成功获取锁,我们将调度一个排队的作业。 此外,我们将通过锁的php owner
方法将锁的所有者令牌传递给排队的作业:
$podcast = Podcast::find($id);
$lock = Cache::lock('processing', 120);
if ($lock->get()) {
ProcessPodcast::dispatch($podcast, $lock->owner());
}
在我们应用程序的php ProcessPodcast
作业中,我们可以使用所有者令牌恢复和释放锁:
Cache::restoreLock('processing', $this->owner)->release();
如果你想释放一个锁而不考虑它的当前所有者,你可以使用php forceRelease
方法:
Cache::lock('processing')->forceRelease();
添加自定义缓存驱动
编写驱动
要创建我们的自定义缓存驱动程序,我们首先需要实现php Illuminate\Contracts\Cache\Store
contract。 因此,MongoDB 缓存实现可能如下所示:
<?php
namespace App\Extensions;
use Illuminate\Contracts\Cache\Store;
class MongoStore implements Store
{
public function get($key) {}
public function many(array $keys) {}
public function put($key, $value, $seconds) {}
public function putMany(array $values, $seconds) {}
public function increment($key, $value = 1) {}
public function decrement($key, $value = 1) {}
public function forever($key, $value) {}
public function forget($key) {}
public function flush() {}
public function getPrefix() {}
}
我们只需要使用 MongoDB 连接来实现这些方法中的每一个。有关如何实现这些方法的示例,请查看 Laravel 框架源代码中的php Illuminate\Cache\MemcachedStore
。 一旦我们的实现完成,我们可以通过调用php Cache
门面的php extend
方法来完成我们的自定义驱动程序注册:
Cache::extend('mongo', function (Application $app) {
return Cache::repository(new MongoStore);
});
技巧
如果你想知道将自定义缓存驱动程序代码放在哪里,可以在你的php app
目录中创建一个php Extensions
命名空间。 但是请记住,Laravel 没有严格的应用程序结构,你可以根据自己的喜好自由组织应用程序。
注册驱动
要向 Laravel 注册自定义缓存驱动程序,我们将使用php Cache
门面的php extend
方法。 由于其他服务提供者可能会尝试在他们的php boot
方法中读取缓存值,我们将在php booting
回调中注册我们的自定义驱动程序。 通过使用php booting
回调,我们可以确保在应用程序的服务提供者调用php boot
方法之前但在所有服务提供者调用php register
方法之后注册自定义驱动程序。 我们将在应用程序的php App\Providers\AppServiceProvider
类的php register
方法中注册我们的php booting
回调:
<?php
namespace App\Providers;
use App\Extensions\MongoStore;
use Illuminate\Contracts\Foundation\Application;
use Illuminate\Support\Facades\Cache;
use Illuminate\Support\ServiceProvider;
class CacheServiceProvider extends ServiceProvider
{
/**
* 注册任何应用程序服务。
*/
public function register(): void
{
$this->app->booting(function () {
Cache::extend('mongo', function (Application $app) {
return Cache::repository(new MongoStore);
});
});
}
/**
* 引导任何应用程序服务。
*/
public function boot(): void
{
// ...
}
}
传递给php extend
方法的第一个参数是驱动程序的名称。这将对应于php config/cache.php
配置文件中的 php driver
选项。 第二个参数是一个闭包,它应该返回一个php Illuminate\Cache\Repository
实例。闭包将传递一个php $app
实例,它是服务容器的一个实例。
注册扩展程序后,将php config/cache.php
配置文件的php driver
选项更新为扩展程序的名称。
事件
要在每个缓存操作上执行代码,你可以侦听缓存触发的 events 。 通常,你应该将这些事件侦听器放在应用程序的php App\Providers\EventServiceProvider
类中:
use App\Listeners\LogCacheHit;
use App\Listeners\LogCacheMissed;
use App\Listeners\LogKeyForgotten;
use App\Listeners\LogKeyWritten;
use Illuminate\Cache\Events\CacheHit;
use Illuminate\Cache\Events\CacheMissed;
use Illuminate\Cache\Events\KeyForgotten;
use Illuminate\Cache\Events\KeyWritten;
/**
* 应用程序的事件侦听器映射。
*
* @var array
*/
protected $listen = [
CacheHit::class => [
LogCacheHit::class,
],
CacheMissed::class => [
LogCacheMissed::class,
],
KeyForgotten::class => [
LogKeyForgotten::class,
],
KeyWritten::class => [
LogKeyWritten::class,
],
];
本文章首发在 网站上。