php解决缓存击穿的问题

缓存击穿是指缓存中没有的数据,而查询非常频繁的数据,导致大量的请求落到了数据库上,因此很容易导致数据库连接数暴增,甚至导致宕机。

下面是 PHP 解决缓存击穿问题的一般解决方案:

// 获取 Key
$key = 'my_key';

// 根据 Key 从 Redis 中获取数据
$data = $redis->get($key);

// 如果数据不存在,尝试从DB中获取数据
if (!$data) {
    // 尝试获取缓存锁
    $lockKey  = 'my_redis_lock';
    $locked   = $redis->set($lockKey, 1, array('nx', 'ex'=>10));

    // 如果获得缓存锁,再从DB中获取
    if($locked) {
        $data = '查询数据库的结果';

        // 将查询结果写入 Redis 缓存中,并设置永久不过期
        $redis->setex($key, 0, $data);
        // 删除锁
        $redis->del($lockKey);
    }
    // 如果获取锁失败,则说明另一个进程已经重新写入了缓存,使用之前写入的数据即可
    else {
        // 设置一个默认的过期时间,以避免缓存被永久锁定
        $redis->setex($key, 60, 'default value');
        // 等待一段时间之后,重新获取缓存
        sleep(1);
        $data = $redis->get($key);
    }
}

该代码逻辑如下:

1.根据指定 Key 从 Redis 中获取数据;

2.如果数据不存在,设置一个缓存互斥锁,防止大量并发访问 DB,再次从 DB/ 获取数据;

3.如果获取到缓存锁,则从 DB 中获取数据,并将数据写入 Redis 中;

4.如果获取不到缓存锁,则说明其他进程已经获取到了缓存锁,等待一段时间后重试,重新获取数据即可。

解决缓存击穿的方法有很多,以上只是其中一种通用的 PHP 解决方法,实际应用中需要结合具体情况选择合适的解决方案。

以下是基于 Redis 分布式锁和补偿任务的代码示例:

// 获取 Key
$key = 'my_key';

// 根据 Key 从 Redis 中获取数据
$data = $redis->get($key);

if (!$data) {
    // 尝试获取分布式锁
    $lockKey = 'my_redis_lock';
    $locked = false; // 是否已经获得锁

    // 尝试获取锁,并最多等待 3 秒钟
    for ($i = 0; $i < 3; $i++) {
        $locked = $redis->set($lockKey, 1, array('nx', 'ex' => 10));
        if ($locked) {
            break;
        }
        sleep(1);
    }

    if ($locked) {
        // 获取数据库数据
        $data = '查询数据库的结果';

        // 将查询结果写入 Redis 缓存中,并设置永久不过期
        $redis->setex($key, 0, $data);
        // 删除锁
        $redis->del($lockKey);
    } else {
        // 使用补偿任务异步处理缓存
        $taskData = array(
            'key' => $key,
        );

        $result = $redis->lpush('my_redis_queue', json_encode($taskData));
        if (!$result) {
            // 记录错误日志
            error_log('Failed to add task to the task queue!');
        }

        // 返回默认值,避免使用错误的数据
        $data = 'default value';
    }
}

// 返回数据
echo $data;

补偿任务相关的代码如下:

// 任务处理函数,将查询结果写入缓存或数据库
function processTask($redis, $taskData) {
    $key = $taskData['key'];
    $data = '查询数据库的结果';

    // 将查询结果写入 Redis 缓存中,并设置永久不过期
    $redis->setex($key, 0, $data);
}

// 拉取任务队列中的任务,并处理任务
function processTaskQueue($redis) {
    while (true) {
        // 从任务队列中阻塞获取任务,最多等待 10 秒钟
        $taskJson = $redis->brpop('my_redis_queue', 10);

        if (!$taskJson) {
            // 没有获取到任务,则结束循环
            break;
        }

        // 解析任务数据
        $taskData = json_decode($taskJson[1], true);

        // 处理任务
        processTask($redis, $taskData);
    }
}

// 启动后台任务处理进程
function startTaskProcessor($redis) {
    while (true) {
        // 处理任务队列中的任务
        processTaskQueue($redis);

        // 等待一段时间后再继续处理任务
        sleep(1);
    }
}

// 在应用程序启动时启动后台任务处理进程
startTaskProcessor($redis);

以上代码示例中,如果获取缓存的进程都无法获得锁并写入缓存,则使用补偿任务异步处理任务,并在后台异步将查询结果写入缓存。补偿任务的处理逻辑与业务逻辑相似,只是将数据的写入缓存操作改为异步操作,并放入任务队列中。后台任务处理进程会不断地从任务队列中获取任务,并进行处理。

文章来源:刘俊涛的博客 欢迎关注公众号、留言、评论,一起学习。


若有帮助到您,欢迎捐赠支持,您的支持是对我坚持最好的肯定(_)

原文链接:https://www.cnblogs.com/lovebing/p/17223016.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:php解决缓存击穿的问题 - Python技术站

(0)
上一篇 2023年4月17日
下一篇 2023年4月17日

相关文章

  • PHP上传 找不到临时文件夹的解决方法

    当使用 PHP 进行文件上传时,有时可能会遇到 “找不到临时文件夹”的问题。这种情况通常发生在临时文件夹不存在或不能访问的情况下。在这篇文章中,我们将讨论如何解决这个问题。 检查服务器配置 首先,我们需要检查服务器是否正确配置了临时文件夹。可以通过以下方式查看php.ini 文件配置: nano /etc/php/7.4/apache2/php.ini 在 …

    PHP 2023年5月27日
    00
  • PHP5.4起内置web服务器使用方法

    PHP5.4起内置web服务器是一个轻量级的基于命令行的web服务器,它可以方便地在开发过程中测试web应用程序,而不用安装和配置传统的web服务器软件,接下来详细讲解PHP5.4起内置web服务器的使用方法。 启动内置web服务器 要启动内置web服务器,需要在命令行中执行以下命令: php -S localhost:8000 其中,localhost表示…

    PHP 2023年5月23日
    00
  • 正则表达式口诀 正则表达式学习工具

    正则表达式是一种用来匹配字符串的工具,它在文本处理中非常常用。正则表达式口诀是一种线上的学习工具,可以辅助用户学习正则表达式。下面我将详细介绍如何使用这个工具。 正则表达式口诀使用攻略 1. 进入正则表达式口诀官网 进入正则表达式口诀官网,打开页面后可以看到一个包含文字和图示的输入框。 2. 输入要匹配的正则表达式 将想要匹配的正则表达式输入到输入框中,可以…

    PHP 2023年5月28日
    00
  • PHP中__set()实例用法和基础讲解

    PHP中__set()实例用法和基础讲解 1. __set() 方法 __set() 方法用于设置私有属性的值,当使用未定义或不可见的属性进行赋值时,会自动调用该方法。 public void __set ( string $name , mixed $value ) 其中,$name 表示属性名,$value 表示属性值。 2. __set() 方法的实例…

    PHP 2023年5月25日
    00
  • PHP动态输出JavaScript代码实例

    下面是“PHP动态输出JavaScript代码实例”的详细攻略。 概述 在网页中,有时需要根据用户输入的不同来动态输出JavaScript代码。为了实现这个功能,就需要使用PHP来动态生成JavaScript代码。本文将详细介绍如何使用PHP动态输出JavaScript代码。 代码示例一 以一个简单的表单为例,该表单有一个输入框和一个按钮,用户可以在输入框中…

    PHP 2023年5月23日
    00
  • PHP输出JSON格式数据方式

    下面是“PHP输出JSON格式数据方式”的完整使用攻略,包括JSON格式数据的介绍、PHP输出JSON格式数据的方式和两个示例说明。 JSON格式数据介绍 JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,它基于JavaScript语言的一个子集,可以被多种编程语言解析和生成。JSON格式数据具有易读、易写、易解析、…

    PHP 2023年5月12日
    00
  • php常用字符串查找函数strstr()与strpos()实例分析

    PHP常用字符串查找函数:strstr()与strpos()实例分析 简介 在PHP中,字符串是经常出现的数据类型之一。而字符串常用的操作之一就是查找。在查找字符串中是否包含另一个字符串时,PHP提供了两个常用的函数:strstr()和strpos()。 请注意区分,strstr()和strpos()的返回值有所不同,具体解释见下文分析。 下面我们分别介绍这…

    PHP 2023年5月26日
    00
  • php利用嵌套数组拼接与解析json的方法

    PHP利用嵌套数组拼接与解析JSON的方法 什么是嵌套数组 在PHP中,一个数组可以包含多个元素。如果一个元素本身又是一个数组,那么我们就称这个数组为嵌套数组。例如: $array = [ ‘name’ => ‘张三’, ‘age’ => 20, ‘hobbies’ => [ ‘篮球’, ‘足球’, ‘游泳’ ] ]; 在上面的代码中,$a…

    PHP 2023年5月26日
    00
合作推广
合作推广
分享本页
返回顶部