PHP使用复述,作为缓存时怎么实现高效缓存

  介绍

本篇内容介绍了“PHP使用复述,作为缓存时怎么实现高效缓存”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

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

<李>

前后台模块共用模型层;

<李>

但是,不能每个模型类都进行缓存,这样太浪费复述,资源;

<李>

前后台模块可以自由决定从数据库还是从缓存读数据;

<李>

没有冗余代码;

<李>

使用方便。
这里我们先展示实现的最终效果。

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

马上安装使用命令:

<>之前,美元composer  install  yeszao/缓存

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

<强> 1最终效果

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

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

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

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

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

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

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

<强> 2 __call()方法

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

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

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

假设我们有一个书类:

 class 书
  {
  ,,,public  function  __call(名称、美元,美元参数)
  ,,,{
  ,,,,,,,echo  & # 39;类的书不存在方法& # 39;,,名字,美元,PHP_EOL;
  ,,,}
  
  ,,,public  function  getById ($ id)
  ,,,{
  ,,,,,,,echo  & # 39;我的ID是& # 39;,,$ ID, PHP_EOL;
  ,,,}
  }

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

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

这就是__call的原理。

<强> 3实现细节

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

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

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

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

 class 书
  {
  ,,,public  function  __call(名称、美元,美元参数)
  ,,,{
  ,,,,,,,//,待填充内容
  ,,,}
  
  ,,,public  function  getById ($ id)
  ,,,{
  ,,,,,,,return  [& # 39; id # 39;,=祝辞,$ id, & # 39;标题# 39;,=祝辞,& # 39;PHP缓存技术& # 39;,只$ id];
  ,,,}
  }

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

 class 常见
  {
  ,,,private  static  redis 美元;=,空;
  
  ,,,public  static  function 复述()
  ,,,{
  ,,,,,,,if  (self:: $ redis ===, null), {
  ,,,,,,,,,,,self:: $ redis =, new  \复述(& # 39;127.0.0.1 # 39;);
  ,,,,,,,,,,,self:: $复述→连接(& # 39;复述,# 39;);
  ,,,,,,,}
  ,,,,,,,return 自我::$复述;
  }

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

 class 书
  {
  ,,,public  function  __call(名称、美元,美元参数)
  ,,,{
  ,,,,,,,//,因为我们主要是根据方法名的后缀决定具体操作,
  ,,,,,,,//,所以如果传入的,name 美元;长度小于5日可以直接报的错
  ,,,,,,,if  (strlen(名字),& lt;, 5),{退出(& # 39;Method  does  not 强生存在# 39;);}//,接着,我们截取,美元的名字,获取原方法和要执行的动作,//,是缓存,明显还是冲洗,这里我们取了个巧,动作//,的名称都是5个字符,这样截取就非常高效。Method 美元;=,substr(名称、美元,0,,5);action 美元;=,substr(名称、美元,5);//,当前调用的类名称,包括命名空间的名称class 美元;=,get_class();//,生成缓存键名,参数稍美元后再加上key 美元;=,sprintf (& # 39; % s: % s: & # 39;,, str_replace(& # 39; \ \ & # 39;,, & # 39; _ # 39;,,美元类),,美元的方法);//,都用小写好看点key 美元;=,函数(键)美元;switch (动作),{case  & # 39;缓存# 39;://,缓存键名加上争论key 美元=,key 美元;只md5 (json_encode(参数)美元);//,从复述中读取数据data 美元;=,常见的::复述()→(键);
  
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null
  null

PHP使用复述,作为缓存时怎么实现高效缓存