是否想过PHP使用redis作为缓存时,如何能:

  1. 前后台模块共用Model层;
  2. 但是,不能每个Model类都进行缓存,这样太浪费Redis资源;
  3. 前后台模块可以自由决定从数据库还是从缓存读数据;
  4. 没有冗余代码;
  5. 使用方便。
    这里我们先展示实现的最终效果。

最终的代码和使用说明请移步Github:https://github.com/yeszao/php-redis-cache

马上安装使用命令:

$ composer install yeszao/cache

经过简单配置就可以使用,请参看Github的README说明。

1 最终效果

假设在MVC框架中,model层有一个Book类和一个getById方法,如下:

class Book
{
public function getById($id)
{
return $id;
}
}

加入缓存技术之后,原来方法的调用方式和返回的数据结构都不应该改变。

所以,我们希望,最后的效果应该是这样的:

 (new Book)->getById();           // 原始的、不用缓存的调用方式,还是原来的方式,一般是读取数据库的数据。
(new Book)->getByIdCache(); // 使用缓存的调用方式,缓存键名为:app_models_book:getbyid: + md5(参数列表)
(new Book)->getByIdClear(); // 删除这个缓存
(new Book)->getByIdFlush(); // 删除 getById() 方法对应的所有缓存,即删除 app_models_book:getbyid:*。这个方法不需要参数。

这样我们可以很清楚的明白自己在做什么,同时又知道数据的来源函数,并且被引用方式完全统一,可谓一箭三雕。

其实实现起来也比较简单,就是使用PHP的魔术方法__call()方法。

2 __call()方法

这里简单说明一下__call方法的作用。

在PHP中,当我们访问一个不存在的类方法时,就会调用这个类的__call()方法。

(如果类方法不存在,又没有写__call()方法,PHP会直接报错)

假设我们有一个Book类:

 class Book
{
public function __call($name, $arguments)
{
echo '类Book不存在方法', $name, PHP_EOL;
} public function getById($id)
{
echo '我的ID是', $id, PHP_EOL;
}
}

当调用存在的getById(50)方法时,程序打印:我的ID是50。

而如果调用不存在的getAge()方法时,程序就会执行到A类的__call()方法里面,这里会打印:类Book不存在方法getAge。

这就是__call的原理。

3 实现细节

接下来我们就利用__call()方法的这种特性,来实现缓存策略。

从上面的例子,我们看到,__call()方法被调用时,会传入两个参数。

name:想要调用的方法名arguments:参数列表
我们就可以在参数上面做文章。

还是以Book类为例,我们假设其原本结构如下:

 class Book
{
public function __call($name, $arguments)
{
// 待填充内容
} public function getById($id)
{
return ['id' => $id, 'title' => 'PHP缓存技术' . $id];
}
}

开始之前,我们还确认Redis的连接,这是缓存必须用到的,这里我们写个简单的单例类:

 class Common
{
private static $redis = null; public static function redis()
{
if (self::$redis === null) {
self::$redis = new \Redis('127.0.0.1');
self::$redis->connect('redis');
}
return self::$redis;
}

然后,我们开始填充__call()方法代码,具体说明请看注释:

 class Book
{
public function __call($name, $arguments)
{
// 因为我们主要是根据方法名的后缀决定具体操作,
// 所以如果传入的 $name 长度小于5,可以直接报错
if (strlen($name) < ) {
exit('Method does not exist.');
} // 接着,我们截取 $name,获取原方法和要执行的动作,
// 是cache、clear还是flush,这里我们取了个巧,动作
// 的名称都是5个字符,这样截取就非常高效。
$method = substr($name, , -);
$action = substr($name, -); // 当前调用的类名称,包括命名空间的名称
$class = get_class(); // 生成缓存键名,$arguments稍后再加上
$key = sprintf('%s:%s:', str_replace('\\', '_', $class), $method);
// 都用小写好看点
$key = strtolower($key); switch ($action) {
case 'Cache':
// 缓存键名加上$arguments
$key = $key . md5(json_encode($arguments)); // 从Redis中读取数据
$data = Common::redis()->get($key); // 如果Redis中有数据
if ($data !== false) {
$decodeData = json_decode($data, JSON_UNESCAPED_UNICODE);
// 如果不是JSON格式的数据,直接返回,否则返回json解析后的数据
return $decodeData === null ? $data : $decodeData;
} // 如果Redis中没有数据则继续往下执行 // 如果原方法不存在
if (method_exists($this, $method) === false) {
exit('Method does not exist.');
} // 调用原方法获取数据
$data = call_user_func_array([$this, $method], $arguments); // 保存数据到Redis中以便下次使用
Common::redis()->set($key, json_encode($data), ); // 结束执行并返回数据
return $data;
break; case 'Clear':
// 缓存键名加上$arguments
$key = $key . md5(json_encode($arguments));
return Common::redis()->del($key);
break; case 'Flush':
$key = $key . '*'; // 获取所有符合 $class:$method:* 规则的缓存键名
$keys = Common::redis()->keys($key);
return Common::redis()->del($keys);
break; default:
exit('Method does not exist.');
}
} // 其他方法
}

这样就实现了我们开始时的效果。

4 实际使用时

在实际使用中,我们需要做一些改变,把这一段代码归入一个类中,

然后在model层的基类中引用这个类,再传入Redis句柄、类对象、方法名和参数,

这样可以降低代码的耦合,使用起来也更灵活。

完整的代码已经放在Github上,请参考文章开头的参考地址。

推荐阅读:

PHP操作Redis数据库常用方法

Redis的面试问题总结,面试跳槽必备

用PHP+Redis实现延迟任务,实现自动取消订单

PHP基于Redis实现轻量级延迟队列

php+redis实现注册、删除、编辑、分页、登录、关注等功能

PHP 面试官问:你说说Redis的几个过期策略?

高效PHP Redis缓存技术,可参考下步骤的更多相关文章

  1. redis缓存技术

    初学redis缓存技术,如果文章写得不好还请谅解 应用环境:win7 实现环境:cmd,eclipse redis缓存技术的特点就在于高效,因为目前涉及的数据量逐渐增多,在对于数据的存储上面和sql以 ...

  2. redis缓存技术学习

    1 什么是redis redis是一个key-value存储系统.和Memcached类似,它支持存储的value类型相对更多,包括string(字符串). list(链表).set(集合)和zset ...

  3. C#版-Redis缓存服务器在Windows下的使用

    Redis缓存服务器是一款key/value数据库,读110000次/s,写81000次/s,因为是内存操作所以速度飞快,常见用法是存用户token.短信验证码等 官网显示Redis本身并没有Wind ...

  4. redis 缓存技术与memcache的区别

    1 什么是redis redis是一个key-value存储系统.和Memcached类似,它支持存储的value类型相对更多,包括string(字符串).list(链表).set(集合)和zset( ...

  5. redis 缓存技术与memcache的最大差别

    1 什么是redis  redis是一个key-value存储系统.和Memcached类似,它支持存储的value类型相对很多其它,包含string(字符串).list(链表).set(集合)和zs ...

  6. redis缓存技术在java中的应用

    https://blog.csdn.net/zhh1072773034/article/details/78538012

  7. php缓存技术有哪些(总结)

    php缓存技术有哪些(总结) 一.总结 一句话总结: 静态页面:全页面静态化缓存,页面部分缓存(将页面中不常变动的部分进行静态化缓存), 数据缓存:比如我的每轮的题目数据,商店,寻宝数据等 数据库:查 ...

  8. 面试官:你对Redis缓存了解吗?面对这11道面试题你是否有很多问号?

    前言 关于Redis的知识,总结了一个脑图分享给大家 1.在项目中缓存是如何使用的?为什么要用缓存?缓存使用不当会造成什么后果? 面试官心理分析 这个问题,互联网公司必问,要是一个人连缓存都不太清楚, ...

  9. redis缓存使用详解

    mysql数据库是存在磁盘中的,操作是对于磁盘操作,这样访问量和并发很大时,运行速率就取决于磁盘的容量,带宽的大小和读取的方式,也就是 sql 语句,次数和效率也会影响读取效率.当访问量和并发很大的时 ...

随机推荐

  1. Dockerfile 指令详解

    GitHub Page:https://blog.cloudli.top/posts/Dockerfile-指令详解/ FROM FROM 命令指定基础镜像.在构建镜像时,基础镜像必须指定,因此在 D ...

  2. Spring为IOC容器注入Bean的五种方式

    一 @Import导入组件,id默认是组件的全类名 //类中组件统一设置.满足当前条件,这个类中配置的所有bean注册才能生效: @Conditional({WindowsCondition.clas ...

  3. Java内存模型(JMM)详解

    在Java JVM系列文章中有朋友问为什么要JVM,Java虚拟机不是已经帮我们处理好了么?同样,学习Java内存模型也有同样的问题,为什么要学习Java内存模型.它们的答案是一致的:能够让我们更好的 ...

  4. python argparse:命令行参数解析详解

    简介 本文介绍的是argparse模块的基本使用方法,尤其详细介绍add_argument内建方法各个参数的使用及其效果. 本文翻译自argparse的官方说明,并加上一些笔者的理解 import a ...

  5. 中文预训练模型ERNIE2.0模型下载及安装

    2019年7月,百度ERNIE再升级,发布持续学习的语义理解框架ERNIE 2.0,及基于此框架的ERNIE 2.0预训练模型, 它利用百度海量数据和飞桨(PaddlePaddle)多机多卡高效训练优 ...

  6. MIT线性代数:6.列向量和零空间

  7. NFS共享目录

    NFS(Network Files System)即网络文件系统 NFS文件系统协议允许网络中的主机通过TCP/IP协议进行资源共享,NFS客户端可以像使用本地资源一样读写远端NFS服务端的资料,需要 ...

  8. 201871010114-李岩松《面向对象程序设计(java)》第一周学习总结

    项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ 这个作业的要求在哪里 https://www.cnblogs.com/nwnu-daizh/p ...

  9. 实验:基于http的yum源

    实验:基于http的yum源 selinux,firewalld已经关闭',系统为CentOS7 repodata所在的目录就是yum源 下面介绍了如何把本地光盘通过httpd服务器变成yum源:多个 ...

  10. X-Admin&ABP框架开发-RBAC

    在业务系统需求规划过程中,通常对于诸如组织机构.用户和角色等这种基础功能,通常是将这部分功能规划到通用子域中,这也说明了,对于这部分功能来讲,是系统的基石,整个业务体系是建立于这部分基石之上的,当然, ...