PHP+Redis缓存技术一览

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

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

1.前后台模块共用Model层;

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

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

4. 没有冗余代码;

5. 使用方便。

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

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

https://github.com/yeszao/php-redis-cache
  • 1.

马上安装使用命令:

$ composer install yeszao/cache
  • 1.

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

1、最终效果

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

class Book

{
    public function getById($id)
{
        return $id;
    }
}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.

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

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

(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:*。这个方法不需要参数。
  • 1.
  • 2.
  • 3.
  • 4.

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

其实实现起来也比较简单,就是使用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;
    }
}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.

当调用 存在的 getName(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];
    }
}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.

开始之前,我们还确认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;
}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.

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

class Book
{
    public function __call($name, $arguments)
{
        // 因为我们主要是根据方法名的后缀决定具体操作,
        // 所以如果传入的 $name 长度小于5,可以直接报错
       if (strlen($name) < 5) {
            exit('Method does not exist.');
        }
        // 接着,我们截取 $name,获取原方法和要执行的动作,
        // 是cache、clear还是flush,这里我们取了个巧,动作

        // 的名称都是5个字符,这样截取就非常高效。
        $method = substr($name, 0, -5);
        $action = substr($name, -5);
        // 当前调用的类名称,包括命名空间的名称
        $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), 3600);

                // 结束执行并返回数据
                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.');
        }
    }

    // 其他方法
}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.
  • 48.
  • 49.
  • 50.
  • 51.
  • 52.
  • 53.
  • 54.
  • 55.
  • 56.
  • 57.
  • 58.
  • 59.
  • 60.
  • 61.
  • 62.
  • 63.
  • 64.
  • 65.
  • 66.
  • 67.
  • 68.
  • 69.
  • 70.
  • 71.
  • 72.

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

4、实际使用时

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

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

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

责任编辑:张燕妮 来源: PHP开源社区
相关推荐

2009-12-09 13:47:49

PHP Zend框架模

2009-07-07 10:10:05

PHP开源建站程序

2009-12-08 13:54:31

PHP时间戳函数

2009-12-08 17:01:01

PHP PEAR DB

2009-07-06 00:29:01

开源PHP

2009-08-17 17:19:00

ASP.NET缓存数据

2012-02-27 16:44:01

redisNoSQL

2021-04-07 10:13:51

人工智能深度学习

2009-03-03 20:44:06

桌面虚拟化Xendesktop虚拟化

2010-10-14 16:55:00

MySQL联结查询

2017-03-06 16:34:12

虚拟个人助理

2020-02-17 15:29:00

石墨文档

2021-06-08 09:47:44

Java面向对象

2010-11-15 09:55:35

Oracle转换函数

2011-01-11 09:53:28

linux进程

2011-01-11 10:06:14

linux进程

2019-04-26 14:21:34

手机色彩苹果

2023-11-08 07:45:47

Spring微服务

2016-01-07 13:19:21

大数据分析生态圈

2010-10-21 15:40:05

SQL Server服
点赞
收藏

51CTO技术栈公众号