您现在的位置是:群英 > 开发技术 > PHP语言
PHP使用redis作为缓存的相关问题有哪些
Admin发表于 2022-10-08 17:52:20385 次浏览
今天这篇给大家分享的知识是“PHP使用redis作为缓存的相关问题有哪些”,小编觉得挺不错的,对大家学习或是工作可能会有所帮助,因此分享发大家做个参考,下文的讲解详细,步骤过程清晰,希望这篇“PHP使用redis作为缓存的相关问题有哪些”文章能帮助大家解决问题。

 


有否想过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(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类:

class Book
{
    public function __call($name, $arguments)
    {
        echo '类Book不存在方法', $name, PHP_EOL;
    }

    public function getById($id)
    {
        echo '我的ID是', $id, PHP_EOL;
    }
}

当调用存在的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];
    }
}

开始之前,我们还确认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) < 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.');
        }
    }

    // 其他方法
}

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

4 实际使用时

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

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

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


关于“PHP使用redis作为缓存的相关问题有哪些”的内容就介绍到这,感谢各位的阅读,相信大家对PHP使用redis作为缓存的相关问题有哪些已经有了进一步的了解。大家如果还想学习更多知识,欢迎关注群英网络,小编将为大家输出更多高质量的实用文章!

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:mmqy2019@163.com进行举报,并提供相关证据,查实之后,将立刻删除涉嫌侵权内容。

标签: PHP使用redis
相关信息推荐
2022-10-14 17:56:28 
摘要:这篇文章主要介绍了springBoot项目中使用@Value取值出现的问题及解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2022-10-13 17:41:10 
摘要:c编译程序又称c语言编译器,是指用c语言书写的源程序,翻译成等价的机器语言格式目标程序的翻译程序。c编译程序首先会检查源程序的正确性,并把它分解成若干基本成分;然后根据这些基本成分建立相应等价的目标程序部分。
2022-06-28 17:04:59 
摘要:有一种说法,golang 编写的 API 不能像其他语言那样简单和通用。但实际上,我遇到很多 REST API 的代码,非常多的抽象,使得代码库变得混乱和复杂,最终伤害了可读性和可维护性。
云活动
推荐内容
热门关键词
热门信息
群英网络助力开启安全的云计算之旅
立即注册,领取新人大礼包
  • 联系我们
  • 24小时售后:4006784567
  • 24小时TEL :0668-2555666
  • 售前咨询TEL:400-678-4567

  • 官方微信

    官方微信
Copyright  ©  QY  Network  Company  Ltd. All  Rights  Reserved. 2003-2019  群英网络  版权所有   茂名市群英网络有限公司
增值电信经营许可证 : B1.B2-20140078   粤ICP备09006778号
免费拨打  400-678-4567
免费拨打  400-678-4567 免费拨打 400-678-4567 或 0668-2555555
微信公众号
返回顶部
返回顶部 返回顶部