0

0

PHP缓存数据怎样防雪崩效应_PHP防雪崩缓存法【防护】

看不見的法師

看不見的法師

发布时间:2026-02-07 13:08:04

|

299人浏览过

|

来源于php中文网

原创

缓存雪崩是大量缓存集体失效导致数据库瞬间过载;PHP中表现为Redis::get大量返回null后mysqli_query/PDO::query出现连接超时、CPU飙升、慢日志暴增,常见于定时刷新或同TTL设置场景。

php缓存数据怎样防雪崩效应_php防雪崩缓存法【防护】

缓存雪崩是什么,PHP里怎么一眼识别

缓存雪崩不是慢,是集体失效后数据库被瞬间打穿。在 PHP 中典型表现是:Redis::get() 大量返回 null,紧接着 mysqli_query()PDO::query() 出现连接超时、CPU 突升、慢日志暴增。尤其发生在定时任务刷新缓存后、或大量 key 设置了相同过期时间(比如都设 3600 秒)的场景。

给缓存加随机过期时间(最简单有效的防线)

别让所有 key 在同一秒过期。PHP 中设置缓存时,在基础 TTL 上叠加一个随机偏移量:

```php
$baseTtl = 3600; // 1小时
$randomOffset = rand(0, 600); // ±10分钟
$ttl = $baseTtl + $randomOffset;

$redis->setex($key, $ttl, $value);

  • 偏移量建议控制在基础 TTL 的 10%–20%,太小没用,太大影响一致性
  • 不要用 time() + $ttl 算过期时间再存——setex / setEX 参数才是原子性保障
  • 如果用 apcu_store(),也得手动加随机: apcu_store($key, $value, $ttl + rand(0, 300))

用互斥锁(mutex)挡住并发回源请求

当缓存失效时,多个 PHP-FPM 进程同时发现没数据,都会去查 DB——这就是“击穿+雪崩”的温床。必须只放行一个进程重建缓存:

Face++旷视
Face++旷视

Face⁺⁺ AI开放平台

下载

立即学习PHP免费学习笔记(深入)”;

```php
$key = 'user:123:profile';
$lockKey = $key . ':lock';
$lockExpire = 30; // 锁最多 hold 30 秒,防死锁

// 尝试加锁
if ($redis->set($lockKey, 1, ['NX', 'EX' => $lockExpire])) {
    try {
        $data = $pdo->query("SELECT * FROM users WHERE id = 123")->fetch();
        $redis->setex($key, 3600 + rand(0, 600), json_encode($data));
    } finally {
        $redis->del($lockKey); // 必须释放
    }
} else {
    // 等待 100ms 后重试(或直接 sleep(0.1)),避免忙等
    usleep(100000);
    $value = $redis->get($key); // 再捞一次,可能已被别的进程写入
}
```
  • NX 是关键:只有 key 不存在才设成功,保证锁唯一性
  • 锁过期时间必须短于业务查询最大耗时,否则会误删别人持有的锁
  • 别用 file_put_contents(..., LOCK_EX) 做锁——跨机器不生效,且性能差

兜底:本地缓存 + 熔断降级(PHP 层最后一道闸)

Redis 挂了或网络抖动时,不能全量回源。PHP 应保留一层轻量本地缓存(如 APCu)并配合简单熔断逻辑:

```php
function getCachedUser($id) {
    $key = "user:{$id}";

    // 先查 APCu(本地,无网络开销)
    if (($local = apcu_fetch($key)) !== false) {
        return $local;
    }

    // 再查 Redis
    $redisValue = $redis->get($key);
    if ($redisValue !== false) {
        apcu_store($key, $redisValue, 60); // 本地缓存 60 秒,减轻 Redis 压力
        return $redisValue;
    }

    // 缓存未命中 → 触发熔断检查
    if (isCircuitBreakerOpen('user_db')) {
        return ['name' => '用户信息暂不可用', 'offline_fallback' => true];
    }

    // 查库 + 写两级缓存
    $row = $pdo->query("...")->fetch();
    $redis->setex($key, 3600 + rand(0, 600), json_encode($row));
    apcu_store($key, json_encode($row), 60);
    return $row;
}
```
  • APCu 不是万能的,它只在单 FPM 进程内有效,所以不能替代分布式缓存,但能显著缓解 Redis 故障时的冲击
  • 熔断器状态建议存在 Redis 里(带过期),由所有进程共享;简单实现可用 redis->incr($breakerKey) + 计数窗口
  • fallback 数据结构要和正常数据保持兼容,否则调用方容易报错
实际部署中,随机 TTL 和 mutex 是必选项,APCu + 熔断看业务容忍度。最容易被忽略的是锁的过期时间与业务查询耗时的匹配——设短了会误删,设长了卡住后续请求,这个值必须实测。

相关文章

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

241

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

601

2024.03.01

php连接mssql数据库的方法
php连接mssql数据库的方法

php连接mssql数据库的方法有使用PHP的MSSQL扩展、使用PDO等。想了解更多php连接mssql数据库相关内容,可以阅读本专题下面的文章。

1629

2023.10.23

常用的数据库软件
常用的数据库软件

常用的数据库软件有MySQL、Oracle、SQL Server、PostgreSQL、MongoDB、Redis、Cassandra、Hadoop、Spark和Amazon DynamoDB。更多关于数据库软件的内容详情请看本专题下面的文章。php中文网欢迎大家前来学习。

986

2023.11.02

内存数据库有哪些
内存数据库有哪些

内存数据库有Redis、Memcached、Apache Ignite、VoltDB、TimesTen、H2 Database、Aerospike、Oracle TimesTen In-Memory Database、SAP HANA和ache Cassandra。更多关于内存数据库相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

650

2023.11.14

mongodb和redis哪个读取速度快
mongodb和redis哪个读取速度快

redis 的读取速度比 mongodb 更快。原因包括:1. redis 使用简单的键值存储,而 mongodb 存储 json 格式的数据,需要解析和反序列化。2. redis 使用哈希表快速查找数据,而 mongodb 使用 b-tree 索引。因此,redis 在需要高性能读取操作的应用程序中是一个更好的选择。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

493

2024.04.02

redis怎么做缓存服务器
redis怎么做缓存服务器

redis 作为缓存服务器的答案:redis 是一款开源、高性能、分布式的键值存储,可作为缓存服务器使用。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

404

2024.04.07

redis怎么解决数据一致性
redis怎么解决数据一致性

redis 提供了两种一致性模型,以维护副本数据一致性:强一致性 (sync) 确保写操作仅在复制到所有从节点后才完成;最终一致性 (async) 则在主节点上写操作后认为已完成,牺牲一致性换取性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

399

2024.04.07

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

2

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 2.1万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 823人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号