当前位置:首页 > PHP教程 > php高级应用 > 列表

PHP使用redis作为缓存(高效技术)

发布:smiling 来源: PHP粉丝网  添加日期:2022-05-28 09:44:31 浏览: 评论:0 

高效PHP Redis缓存技术,可参考下步骤

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

● 前后台模块共用Model层;

● 但是,不能每个Model类都进行缓存,这样太浪费Redis资源;

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

● 没有冗余代码;

● 使用方便。

● 这里我们先展示实现的最终效果。

最终的代码和使用说明请移步Github:

https://github.com/yeszao/php-redis-cache。

马上安装使用命令:

$ composer install yeszao/cache

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

1 最终效果

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

  1. class Book 
  2.  
  3.  
  4.     public function getById($id
  5.  
  6.     { 
  7.  
  8.         return $id
  9.  
  10.     } 
  11.  

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

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

(new Book)->getById(100);           // 原始的、不用缓存的调用方式,还是原来的方式,一般是读取数据库的数据。

(new Book)->getByIdCache(100);      // 使用缓存的调用方式,缓存键名为:app_models_book:getbyid: + md5(参数列表)

(new Book)->getByIdClear(100);      // 删除这个缓存

(new Book)->getByIdFlush();         // 删除 getById() 方法对应的所有缓存,即删除 app_models_book:getbyid:*。这个方法不需要参数。

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

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

2 __call()方法

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

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

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

假设我们有一个Book类:

  1. class Book 
  2.  
  3.  
  4.     public function __call($name$arguments
  5.  
  6.     { 
  7.  
  8.         echo '类Book不存在方法'$name, PHP_EOL; 
  9.  
  10.     } 
  11.  
  12.  
  13.  
  14.     public function getById($id
  15.  
  16.     { 
  17.  
  18.         echo '我的ID是'$id, PHP_EOL; 
  19.  
  20.     } 
  21.  

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

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

这就是__call的原理。

3 实现细节

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

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

name:想要调用的方法名arguments:参数列表

我们就可以在参数上面做文章。

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

  1. class Book 
  2.  
  3.  
  4.     public function __call($name$arguments
  5.  
  6.     { 
  7.  
  8.         // 待填充内容 
  9.  
  10.     } 
  11.  
  12.  
  13.  
  14.     public function getById($id
  15.  
  16.     { 
  17.  
  18.         return ['id' => $id'title' => 'PHP缓存技术' . $id]; 
  19.  
  20.     } 
  21.  

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

  1. class Common 
  2.  
  3.  
  4.     private static $redis = null; 
  5.  
  6.  
  7.  
  8.     public static function redis() 
  9.  
  10.     { 
  11.  
  12.         if (self::$redis === null) { 
  13.  
  14.             self::$redis = new \Redis('127.0.0.1'); 
  15.  
  16.             self::$redis->connect('redis'); 
  17.  
  18.         } 
  19.  
  20.         return self::$redis
  21.  

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

  1. class Book 
  2.  
  3.  
  4.     public function __call($name$arguments
  5.  
  6.     { 
  7.  
  8.         // 因为我们主要是根据方法名的后缀决定具体操作, 
  9.  
  10.         // 所以如果传入的 $name 长度小于5,可以直接报错 
  11.  
  12.         if (strlen($name) < 5) { 
  13.  
  14.             exit('Method does not exist.'); 
  15.  
  16.         } 
  17.  
  18.  
  19.  
  20.         // 接着,我们截取 $name,获取原方法和要执行的动作, 
  21.  
  22.         // 是cache、clear还是flush,这里我们取了个巧,动作 
  23.  
  24.         // 的名称都是5个字符,这样截取就非常高效。 
  25.  
  26.         $method = substr($name, 0, -5); 
  27.  
  28.         $action = substr($name, -5); 
  29.  
  30.  
  31.  
  32.         // 当前调用的类名称,包括命名空间的名称 
  33.  
  34.         $class = get_class(); 
  35.  
  36.  
  37.  
  38.         // 生成缓存键名,$arguments稍后再加上 
  39.  
  40.         $key = sprintf('%s:%s:'str_replace('\\', '_', $class), $method); 
  41.  
  42.         // 都用小写好看点 
  43.  
  44.         $key = strtolower($key); 
  45.  
  46.  
  47.  
  48.         switch ($action) { 
  49.  
  50.             case 'Cache'
  51.  
  52.                 // 缓存键名加上$arguments 
  53.  
  54.                 $key = $key . md5(json_encode($arguments)); 
  55.  
  56.  
  57.  
  58.                 // 从Redis中读取数据 
  59.  
  60.                 $data = Common::redis()->get($key); 
  61.  
  62.  
  63.  
  64.                 // 如果Redis中有数据 
  65.  
  66.                 if ($data !== false) { 
  67.  
  68.                     $decodeData = json_decode($data, JSON_UNESCAPED_UNICODE); 
  69.  
  70.                     // 如果不是JSON格式的数据,直接返回,否则返回json解析后的数据 
  71.  
  72.                     return $decodeData === null ? $data : $decodeData
  73.  
  74.                 } 
  75.  
  76.  
  77.  
  78.                 // 如果Redis中没有数据则继续往下执行 
  79.  
  80.  
  81.  
  82.                 // 如果原方法不存在 
  83.  
  84.                 if (method_exists($this$method) === false) { 
  85.  
  86.                     exit('Method does not exist.'); 
  87.  
  88.                 } 
  89.  
  90.  
  91.  
  92.                 // 调用原方法获取数据 
  93.  
  94.                 $data = call_user_func_array([$this$method], $arguments); 
  95.  
  96.  
  97.  
  98.                 // 保存数据到Redis中以便下次使用 
  99.  
  100.                 Common::redis()->set($key, json_encode($data), 3600); 
  101.  
  102.  
  103.  
  104.                 // 结束执行并返回数据 
  105.  
  106.                 return $data
  107.  
  108.                 break
  109.  
  110.  
  111.  
  112.             case 'Clear'
  113.  
  114.                 // 缓存键名加上$arguments 
  115.  
  116.                 $key = $key . md5(json_encode($arguments)); 
  117.  
  118.                 return Common::redis()->del($key); 
  119.  
  120.                 break
  121.  
  122.  
  123.  
  124.             case 'Flush'
  125.  
  126.                 $key = $key . '*'
  127.  
  128.  
  129.  
  130.                 // 获取所有符合 $class:$method:* 规则的缓存键名  
  131.  
  132.                 $keys = Common::redis()->keys($key); 
  133.  
  134.                 return Common::redis()->del($keys); 
  135.  
  136.                 break
  137.  
  138.  
  139.  
  140.             default
  141.  
  142.                 exit('Method does not exist.'); 
  143.  
  144.         } 
  145.  
  146.     } 
  147.  
  148.  
  149.  
  150.     // 其他方法 
  151.  

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

4 实际使用时

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

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

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

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

Tags: redis PHP缓存

分享到: