PHP连接Redis失败时,先确认redis扩展是否加载:用php -m | grep redis检查,未输出则需安装并重启Web服务;Windows需确保php.ini中extension=php_redis.dll未被注释且路径正确;phpinfo()中查到redis模块才表示生效。

PHP连接Redis失败:extension_loaded('redis')返回false怎么办
Redis扩展没装好,PHP根本没法跟Redis说话。不是配错host或port,是压根没连上“电话线”。
- 先确认扩展是否已加载:
php -m | grep redis,没输出就说明没装 - Linux下常用
sudo apt install php-redis(Debian/Ubuntu)或sudo yum install php-pecl-redis(CentOS/RHEL),装完必须重启Web服务(sudo systemctl restart apache2或sudo systemctl restart php-fpm) - Windows用户注意:别只复制
php_redis.dll,还要检查php.ini里有没有这行:extension=php_redis.dll,且路径正确、没有被;注释掉 -
phpinfo()页面里搜“redis”,看到模块信息才算真正生效;只改了php.ini但没重启服务,extension_loaded('redis')一定返回false
new Redis()后调用set()报错“Redis server went away”
连接建立成功,但写入时断开了——多半是超时或认证没过,不是代码写错了。
- 默认构造不带参数会连
127.0.0.1:6379,如果Redis启用了密码,必须显式传参:$redis = new Redis(); $redis->connect('127.0.0.1', 6379, 2.5); $redis->auth('your_password'); - 第三个参数是超时时间(秒),设太小(如
0.1)在高延迟网络下必断;设太大(如30)又拖慢响应,建议2.5~5之间 - 别在
connect()后漏掉isConnected()检查:if (!$redis->isConnected()) { throw new RuntimeException('Redis not ready'); },否则后续操作全抛异常 - 某些Docker环境里Redis绑定
127.0.0.1,但PHP容器里127.0.0.1指向自己,得换用宿主机IP或Docker网络别名(如host.docker.internal)
缓存失效策略:setex()和set() + expire()有什么实际区别
表面都能设过期时间,但原子性差一毫秒,线上就可能出脏数据。
-
setex($key, $ttl, $value)是原子操作,Redis一次完成写+过期设置;而set($key, $value); expire($key, $ttl)是两次网络往返,中间若进程崩溃或Redis挂了,$key就变成永不过期的“幽灵缓存” - TTL单位统一是秒,别把毫秒当秒传(比如
setex('user:123', 30000, $data)其实是设了3万秒≈8小时,不是30秒) - 如果要用更精细的过期控制(比如带条件的setnx+expire),优先考虑
set($key, $value, ['nx', 'ex' => $ttl])这种数组参数形式,PHPRedis 5.3+支持,语义更清晰 - 批量操作时,
setex()无法管道(pipeline),而set()可以,高并发写场景需权衡原子性与吞吐量
序列化问题:存数组后get()出来是乱码或NULL
Redis只认字符串,PHP原生数组、对象不能直接塞进去,得你自己“打包”。
立即学习“PHP免费学习笔记(深入)”;
- 默认情况下
set()会用serialize(),get()自动unserialize(),但前提是值没被其他语言或命令篡改过;一旦有人用redis-cli set user:123 '{"id":1}'写了个JSON,再用PHPget()就会反序列化失败,返回NULL - 更安全的做法是显式控制序列化方式:
$redis->set($key, json_encode($data));,取的时候json_decode($redis->get($key), true),避免PHP序列化格式泄漏或兼容性风险 - 注意
json_encode()对资源、闭包、循环引用会静默失败,返回false,存进去的是字符串"false",读出来再json_decode()就崩了;务必加判断:if (false === $json = json_encode($data)) { throw new InvalidArgumentException(json_last_error_msg()); } - 小数据用
json,大数据考虑igbinary扩展(需单独安装),它比serialize()体积小、速度快,但跨语言不通用
Redis缓存真正难的不是写几行set()和get(),而是连接生命周期管理、序列化边界、原子性兜底——这些地方一松懈,缓存就从加速器变成故障放大器。











