3分钟短文:说说Laravel通用缓存Cache的使用技巧

简介: 用户缓存 Session 与 全局缓存 Cache

引言

前一期我们介绍了存储用户会话数据到服务器,并通过框架提供的Session类
进行数据读写操作的方法。

图片

但是有些数据,如数据库集合,API接口等等,对于全局的用户来说是无差别的,
这时候我们需要全局缓存以便提供给所有的用户使用。

本期就来说说Cache缓存。

代码时间

laravel在设计上做了很巧妙的构思,将缓存相关的类,设计得有插接件的感觉。
所以,你看到Session Cache Cookie 这些缓存数据类,基本上除了底层的驱动,
数据结构,过期特性等等,都集成了系统数组类Arr的操作方法。所以上述三种缓存在
操作方法上有很多相同之处。

cache缓存的配置文件在 config/cache.php 内,支持的驱动默认是 file
也就是文本文件存储。还有很多内置可选项:

  • file 文件存储
  • apc php内置缓存
  • array 内存变量
  • database 数据库
  • memcached / redis 内存数据库
  • dynamodb aws内存数据库

其中array驱动,是把数据直接存在变量内,放在服务器内存中,一次请求结束,或者php-fpm
闲置超时释放,就会注销掉。其他都提供了持久化的方案。

当然了,Cache缓存提供的数据类型就多了去了,根据驱动的不同,支持的存储结构也不相同。
比如文件类,只能是数组,链表,键值。而redis数据库则可以存储集合。这在使用的时候,
需要格外留意。

根据键名获取一个缓存值操作方法如下:

$users = Cache::get('users');

上面的代码使用了系统启动时注册的全局Cache门面方法,我们也可以从容器内获取一个Cache实例,
比如在路由内使用依赖注入的方式实例化:

Route::get('users', function (Illuminate\Contracts\Cache\Repository $cache) {
    return $cache->get('users');
});

laravel也为我们提供了极为便捷的助手函数 cache() 直接操作缓存。因为封装得过于精巧,
所以助手函数内定然是进行了大多数的情况判断。我们对源码稍作解析,以便明白其处理方法。

$arguments = func_get_args();

if (empty($arguments)) {
    return app('cache');
}

如果助手函数没有传入任何参数,比如这样:

$users = cache();

那就使用应用容器实例化一个 Cache::class 类。可以用于后续的链式方法调用。
接着看源码:

if (is_string($arguments[0])) {
    return app('cache')->get(...$arguments);
}

判断第一个位置参数,如果是字符串,那么推断用户就是要获取的键名。那就实例化Cache类并调用get方法获取缓存值。

if (! is_array($arguments[0])) {
    throw new Exception(
        'When setting a value in the cache, you must pass an array of key / value pairs.'
    );
}

为了排除其他所有情况,此处使用了NOT 逻辑判断。
也就是说,助手函数cache()除了接收字符串和数组,其他一概不处理。

接下来,如果传入的是数组,程序的处理方法:

return app('cache')->put(key($arguments[0]), reset($arguments[0]), $arguments[1] ?? null);

如果是数组,系统推断用户是要通过助手函数设置缓存值了。
那就就用第一个位置参数数组的键作为缓存的键名,数组的第一个值作为缓存的值。
同时查看是否有第二个位置参数,将其作为expired过期时间处理。没有的,则默认为null,永不过期。

$users = cache('key', 'default value');
$users = cache()->get('key', 'default value');

上面是根据键名获取缓存值。正好命中了源码中前两种情况。

$users = cache(['key' => 'value'], $minutes);
$users = cache()->put('key', 'value', $minutes);

写入数据也很简单,传入数组和过期时间,或者实例化之后直接调用put方法。

laravel日期时间处理默认使用Carbon类库,所以缓存也用到了该库,
并且把用法发挥到极致了。

比如我们可能想动态地设置过期时间,不计算时长,能不能设置为 到某个时间点结束
自然是可以的。这样来写代码:

cache()->put('key', 'value', Carbon::now()->addDay());

put方法第三个参数传入一个Carbon对象,就是默认缓存到指定的时间点。
至于时长,laravel帮你算好了。我们只关心这个到期时间点,代码可读性明显增强了许多。

其他方法不一一介绍了,特别说一下有个需求,比如说代码中,我们先查找缓存数据,如果没有就去程序上下文获取。
如果存在,就直接返回缓存数据。

这个逻辑用的很多,laravel提供了一个remember语法糖,让你解决这个操作步骤:

$users = cache()->remember('users', 120, function () {
    return User::all();
});

写在最后

本文通过对缓存助手函数cache()的分析,为大家介绍了Cache类传参的一些关键点。
特别是remember方法,在上下文逻辑不复杂的处理中,可有效减少代码量,非常直观。

Happy coding :-)

我是@程序员小助手,持续分享编程知识,欢迎关注。

相关文章
|
3月前
|
存储 缓存 NoSQL
【Azure Redis 缓存】关于Azure Cache for Redis 服务在传输和存储键值对(Key/Value)的加密问题
【Azure Redis 缓存】关于Azure Cache for Redis 服务在传输和存储键值对(Key/Value)的加密问题
|
3月前
|
缓存 弹性计算 NoSQL
【Azure Redis 缓存 Azure Cache For Redis】Redis连接池
【Azure Redis 缓存 Azure Cache For Redis】Redis连接池
|
3月前
|
缓存 Java
Java本地高性能缓存实践问题之使用Caffeine的Cache接口来查找一个缓存元素的问题如何解决
Java本地高性能缓存实践问题之使用Caffeine的Cache接口来查找一个缓存元素的问题如何解决
|
3月前
|
缓存 NoSQL Java
【Azure Redis 缓存 Azure Cache For Redis】Redis出现 java.net.SocketTimeoutException: Read timed out 异常
【Azure Redis 缓存 Azure Cache For Redis】Redis出现 java.net.SocketTimeoutException: Read timed out 异常
|
3月前
|
存储 缓存 NoSQL
【Azure Redis 缓存 Azure Cache For Redis】如何设置让Azure Redis中的RDB文件暂留更久(如7天)
【Azure Redis 缓存 Azure Cache For Redis】如何设置让Azure Redis中的RDB文件暂留更久(如7天)
|
3月前
|
缓存 NoSQL Redis
【Azure Redis 缓存】Azure Cache for Redis 服务的导出RDB文件无法在自建的Redis服务中导入
【Azure Redis 缓存】Azure Cache for Redis 服务的导出RDB文件无法在自建的Redis服务中导入
|
3月前
|
缓存 开发框架 NoSQL
【Azure Redis 缓存】VM 里的 Redis 能直接迁移到 Azure Cache for Redis ? 需要改动代码吗?
【Azure Redis 缓存】VM 里的 Redis 能直接迁移到 Azure Cache for Redis ? 需要改动代码吗?
|
3月前
|
缓存 NoSQL Unix
【Azure Redis 缓存】Azure Cache for Redis 中如何快速查看慢指令情况(Slowlogs)
【Azure Redis 缓存】Azure Cache for Redis 中如何快速查看慢指令情况(Slowlogs)
|
3月前
|
缓存 NoSQL Redis
【Azure Redis 缓存】Azure Cache for Redis 是否记录具体读/写(Get/Set)或删除(Del)了哪些key呢?
【Azure Redis 缓存】Azure Cache for Redis 是否记录具体读/写(Get/Set)或删除(Del)了哪些key呢?
|
3月前
|
存储 缓存 NoSQL
【Azure Redis 缓存】Azure Cache for Redis 专用终结点, 虚拟网络, 公网访问链路
【Azure Redis 缓存】Azure Cache for Redis 专用终结点, 虚拟网络, 公网访问链路