PHP如何使用Redis缓存_Redis缓存操作完整教程

php如何使用redis缓存_redis缓存操作完整教程

PHP使用Redis缓存的核心在于通过Predisphpredis这样的客户端库,连接到Redis服务器,然后利用其键值存储特性,将需要频繁访问的数据存入内存,以大幅提升应用响应速度。这不仅仅是简单的存取操作,更关乎缓存策略的选择和数据一致性的维护。

解决方案

要在PHP项目中利用Redis进行缓存,我们通常会选择

phpredis

扩展或者

Predis

库。我个人更倾向于在生产环境使用

phpredis

,因为它是一个C扩展,性能上通常会有优势。这里以

phpredis

为例,演示基本操作。

首先,确保你的服务器已经安装了

phpredis

扩展。如果没有,可以通过

pecl install redis

进行安装,并在

php.ini

中启用它。

connect('127.0.0.1', 6379);    // 如果Redis设置了密码,需要进行认证    // $redis->auth('your_redis_password');    echo "成功连接到Redis服务器!n";} catch (RedisException $e) {    die("连接Redis失败: " . $e->getMessage());}// 2. 缓存字符串数据$key = 'my_data_key';$value = 'Hello Redis Cache!';$expireTime = 60; // 缓存60秒if (!$redis->get($key)) { // 检查缓存是否存在    echo "缓存中没有 '{$key}',从数据库或源获取数据并写入缓存...n";    // 模拟从数据库获取数据    $dataFromSource = $value . " (from source)";    $redis->set($key, $dataFromSource, $expireTime); // 设置键值和过期时间    echo "数据已写入缓存: {$dataFromSource}n";} else {    echo "从缓存中获取数据: " . $redis->get($key) . "n";}// 3. 缓存复杂数据类型(例如数组或对象)$complexKey = 'user:1001:profile';$userData = [    'id' => 1001,    'name' => '张三',    'email' => 'zhangsan@example.com',    'roles' => ['admin', 'editor']];// Redis只能存储字符串,所以需要序列化$serializedUserData = json_encode($userData); // 或者使用 serialize()if (!$redis->get($complexKey)) {    echo "缓存中没有 '{$complexKey}',获取用户数据并写入缓存...n";    $redis->set($complexKey, $serializedUserData, 300); // 缓存5分钟    echo "用户数据已写入缓存。n";} else {    $cachedData = $redis->get($complexKey);    $unserializedData = json_decode($cachedData, true); // 或者使用 unserialize()    echo "从缓存中获取用户数据: " . print_r($unserializedData, true) . "n";}// 4. 删除缓存// 假设用户数据更新了,我们需要删除旧缓存// $redis->del($complexKey);// echo "缓存 '{$complexKey}' 已删除。n";// 5. 检查键是否存在if ($redis->exists($key)) {    echo "'{$key}' 键仍然存在于缓存中。n";} else {    echo "'{$key}' 键已过期或不存在。n";}// 6. 设置过期时间(如果之前未设置或需要修改)// $redis->expire($key, 120); // 将 'my_data_key' 的过期时间设置为120秒// 7. 关闭连接 (phpredis会在脚本结束时自动关闭,但显式关闭也是好习惯)$redis->close();?>

这段代码展示了连接Redis、设置带过期时间的缓存、获取缓存、以及处理复杂数据类型的基本流程。实际应用中,你可能需要将这些操作封装成一个服务类,以提高代码的复用性和可维护性。

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

PHP应用中,选择Predis还是phpredis扩展,哪种更适合生产环境?

这个问题,说实话,我个人在项目里遇到过好几次讨论。从纯粹的性能和资源消耗角度看,

phpredis

扩展通常是更优的选择,尤其是在高并发的生产环境中。它是一个用C语言编写的PHP扩展,直接与Redis服务器进行通信,省去了PHP层面的解析和处理开销。这意味着更低的延迟、更高的吞吐量,并且对系统内存的占用也相对较少。安装虽然需要编译,但一旦部署好,维护起来其实很省心。

然而,

Predis

也有其不可替代的优势。它是一个纯PHP的客户端库,这意味着安装极其简单,只需要通过Composer就能引入项目,不需要对PHP环境进行任何编译或配置。这对于开发环境的搭建、或者那些没有权限安装C扩展的共享主机环境来说,简直是福音。此外,

Predis

的代码是纯PHP,对于PHP开发者来说,调试和理解其内部机制也更容易一些。它的API设计也相当现代化,支持各种Redis特性。

所以,我的建议是:

对于绝大多数生产环境,尤其对性能有较高要求的场景,优先考虑

phpredis

它的性能优势是实打实的。如果你的环境限制无法安装C扩展,或者项目初期追求快速搭建,再或者你更看重纯PHP带来的易用性和可调试性,那么

Predis

是一个非常好的替代方案。 很多大型项目也会用

Predis

,通过合理的架构设计,其性能瓶颈往往不在客户端库本身。

最终选择,还得看项目具体需求、团队技术偏好以及部署环境的限制。但如果能用

phpredis

,我通常会毫不犹豫地选择它。

如何设计高效的Redis缓存策略,避免缓存穿透、雪崩与击穿?

设计一个健壮的Redis缓存策略,远不止简单的

set

get

。我们必须警惕“三座大山”:缓存穿透、缓存雪崩和缓存击穿,它们都可能在不经意间把我们的数据库压垮。

缓存穿透 (Cache Penetration)

现象: 查询一个根本不存在的数据,缓存中没有,数据库也没有。每次请求都直接打到数据库,造成数据库压力。恶意攻击者可能会利用这一点。应对策略:缓存空对象/空值: 如果从数据库查询的结果为空,也将其缓存起来(例如,设置一个短时间的空字符串或特定标记),下次查询时直接返回空,避免再次查询数据库。当然,这会占用一些缓存空间,需要权衡。布隆过滤器 (Bloom Filter): 这是一个更高级的方案。在数据写入数据库时,同时将对应的ID(或唯一标识)添加到布隆过滤器中。查询时,先通过布隆过滤器判断该ID是否存在。如果布隆过滤器说不存在,那就一定不存在,直接返回空;如果布隆过滤器说可能存在,再去查缓存和数据库。布隆过滤器有误判率(认为存在但实际不存在),但可以大大减少对数据库的无效查询。

缓存雪崩 (Cache Avalanche)

现象: 大量缓存键在同一时间集体失效,导致所有请求瞬间涌向数据库,数据库扛不住压力而崩溃。这通常发生在设置了相同过期时间的大批热点数据上。应对策略:错开缓存失效时间: 给缓存的过期时间加上一个随机值,例如

expireTime = baseTime + rand(0, 300)

,这样就能让缓存错峰失效,而不是一窝蜂地过期。多级缓存: 引入二级甚至三级缓存。当一级缓存失效时,请求先尝试从二级缓存获取。热点数据永不过期: 对于一些访问频率极高的核心数据,可以考虑将其设置为永不过期,或者在业务低峰期通过后台任务异步刷新缓存。

缓存击穿 (Cache Breakdown)

现象: 某个热点数据缓存失效的瞬间,大量并发请求同时涌入,这些请求都会穿透缓存,直接打到数据库。与雪崩不同,击穿是针对单个热点Key。应对策略:互斥锁 (Mutex): 当一个热点Key失效时,只允许一个请求去查询数据库并重建缓存,其他请求则等待或返回旧数据(如果可以接受)。例如,可以使用Redis的

SETNX

命令来实现分布式锁。

$lockKey = 'lock:' . $hotKey;if ($redis->setnx($lockKey, 1)) { // 尝试获取锁    $redis->expire($lockKey, 10); // 设置锁的过期时间,防止死锁    // 从数据库加载数据,并写入缓存    $data = loadFromDatabase($hotKey);    $redis->set($hotKey, $data, $expireTime);    $redis->del($lockKey); // 释放锁} else {    // 等待或直接返回空/旧数据    usleep(100000); // 等待100ms后重试    return $redis->get($hotKey);}

永不过期 + 异步更新: 将热点数据设置为永不过期,但通过后台线程或定时任务异步地更新缓存。当数据更新时,再将新数据写入缓存。

除了这“三座大山”,还有一些通用的缓存策略建议:

合理设置TTL (Time-To-Live): 根据数据的重要性和更新频率来设置过期时间。不重要的、更新频繁的数据可以设置较短的TTL;重要但更新不频繁的数据可以设置较长的TTL。缓存粒度: 缓存的数据块不宜过大也不宜过小。过大会导致序列化/反序列化开销大,更新困难;过小则会导致Key过多,占用内存,且网络请求频繁。键名设计: 采用统一的命名规范,例如

业务名:表名:ID:字段

,方便管理和查找。

在PHP项目中使用Redis缓存时,常见错误与性能优化技巧有哪些?

在PHP项目里用Redis缓存,虽然能带来巨大的性能提升,但如果不注意一些细节,也容易踩坑或者达不到预期的效果。

常见错误:

不处理缓存异常: 最常见的就是Redis服务挂了,或者网络连接中断,而代码中没有

try-catch

或相应的容错机制。结果就是,整个应用可能因为无法连接Redis而崩溃,或者直接返回错误给用户。建议: 总是用

try-catch

块包裹Redis操作,或者在封装Redis客户端时做好异常处理,确保即使缓存不可用,应用也能降级到直接查询数据库,保证服务的可用性。缓存与数据库数据不一致: 这是缓存最头疼的问题之一。比如,更新了数据库但忘记更新或删除缓存,导致用户看到的是旧数据。建议: 采用“先更新数据库,再删除缓存”的策略(Cache Aside模式)。如果删除缓存失败,可以考虑引入消息队列进行异步重试,或者设置较短的缓存过期时间来降低不一致的窗口。Key命名混乱: 随着项目发展,缓存Key越来越多,如果命名没有规范,很快就会变得难以管理、难以理解,甚至出现Key冲突。建议: 制定严格的Key命名规范,例如

项目名:模块名:业务ID:数据类型

,如

myApp:user:123:profile

缓存粒度不当: 缓存的数据要么太大,导致序列化/反序列化开销大,占用内存多;要么太小,导致Key数量暴增,频繁网络请求,反而降低效率。建议: 结合业务场景,合理划分缓存粒度。例如,一个用户的所有基本信息可以作为一个JSON字符串缓存,而不是每个字段都单独一个Key。不设置过期时间或过期时间过长: 导致Redis内存溢出,或者长时间返回旧数据。建议: 除了极少数需要永不过期的数据,所有缓存都应该设置合理的过期时间。

性能优化技巧:

使用Pipeline (管道) 进行批量操作: 当你需要执行一系列Redis命令时,不要一个接一个地发送请求,而是将它们打包成一个批次,一次性发送给Redis,然后一次性接收所有结果。这能显著减少网络往返时间(RTT),特别是在网络延迟较高的情况下。

$redis->pipeline();$redis->set('key1', 'value1');$redis->set('key2', 'value2');$redis->get('key1');$results = $redis->exec(); // 一次性执行并获取所有结果print_r($results);

利用Lua脚本执行原子操作: 对于一些需要多个步骤才能完成的复杂逻辑(例如“检查库存并扣减”),如果分步执行,可能会因为并发问题导致数据不一致。将这些逻辑封装成Lua脚本,然后通过

EVAL

命令发送给Redis,Redis会保证脚本的原子性执行,避免了竞态条件,同时也减少了网络开销。

$script = "    local current_stock = tonumber(redis.call('get', KEYS[1]))    if current_stock and current_stock >= tonumber(ARGV[1]) then        redis.call('decrby', KEYS[1], ARGV[1])        return 1    end    return 0";// KEYS[1] 是库存key, ARGV[1] 是扣减数量$result = $redis->eval($script, ['product_stock:123', 5], 1);if ($result) {    echo "库存扣减成功!n";} else {    echo "库存不足或操作失败。n";}

选择合适的序列化方式: PHP的

serialize()

函数可以处理各种复杂类型,但其序列化后的字符串通常比

json_encode()

更长,且只能被PHP解析。

json_encode()

生成的JSON字符串更通用,易于跨语言交互,且通常更紧凑。建议: 如果数据需要在PHP之外的其他服务中使用,或者对存储空间和网络传输有要求,优先考虑

json_encode()

。如果只是PHP内部使用且数据结构复杂,

serialize()

也无妨,但要留意其性能开销。合理配置Redis内存和淘汰策略: Redis是内存数据库,如果内存不足,会触发淘汰策略。理解并配置好

maxmemory

maxmemory-policy

(如

allkeys-lru

volatile-lru

等)至关重要,这能确保热点数据被保留,不重要的旧数据被及时淘汰。使用Redis集群或哨兵模式: 对于高可用和横向扩展的需求,单台Redis服务器是不够的。Redis Sentinel (哨兵模式): 提供高可用性,当主节点故障时,自动进行故障转移,选举新的主节点。Redis Cluster (集群模式): 提供数据分片和高可用性,将数据分散到多个节点,实现横向扩展。建议: 在生产环境中,根据业务规模和对可用性、扩展性的要求,选择合适的部署模式。

总的来说,Redis缓存的优化是一个持续的过程,需要结合业务特点、监控数据和实际测试结果来不断调整和完善。它不仅仅是技术问题,更是一门平衡艺术。

以上就是PHP如何使用Redis缓存_Redis缓存操作完整教程的详细内容,更多请关注php中文网其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1320420.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月12日 06:47:00
下一篇 2025年12月12日 06:47:20

相关推荐

  • Uniapp 中如何不拉伸不裁剪地展示图片?

    灵活展示图片:如何不拉伸不裁剪 在界面设计中,常常需要以原尺寸展示用户上传的图片。本文将介绍一种在 uniapp 框架中实现该功能的简单方法。 对于不同尺寸的图片,可以采用以下处理方式: 极端宽高比:撑满屏幕宽度或高度,再等比缩放居中。非极端宽高比:居中显示,若能撑满则撑满。 然而,如果需要不拉伸不…

    2025年12月24日
    400
  • 如何让小说网站控制台显示乱码,同时网页内容正常显示?

    如何在不影响用户界面的情况下实现控制台乱码? 当在小说网站上下载小说时,大家可能会遇到一个问题:网站上的文本在网页内正常显示,但是在控制台中却是乱码。如何实现此类操作,从而在不影响用户界面(UI)的情况下保持控制台乱码呢? 答案在于使用自定义字体。网站可以通过在服务器端配置自定义字体,并通过在客户端…

    2025年12月24日
    800
  • 如何在地图上轻松创建气泡信息框?

    地图上气泡信息框的巧妙生成 地图上气泡信息框是一种常用的交互功能,它简便易用,能够为用户提供额外信息。本文将探讨如何借助地图库的功能轻松创建这一功能。 利用地图库的原生功能 大多数地图库,如高德地图,都提供了现成的信息窗体和右键菜单功能。这些功能可以通过以下途径实现: 高德地图 JS API 参考文…

    2025年12月24日
    400
  • 如何使用 scroll-behavior 属性实现元素scrollLeft变化时的平滑动画?

    如何实现元素scrollleft变化时的平滑动画效果? 在许多网页应用中,滚动容器的水平滚动条(scrollleft)需要频繁使用。为了让滚动动作更加自然,你希望给scrollleft的变化添加动画效果。 解决方案:scroll-behavior 属性 要实现scrollleft变化时的平滑动画效果…

    2025年12月24日
    000
  • 如何为滚动元素添加平滑过渡,使滚动条滑动时更自然流畅?

    给滚动元素平滑过渡 如何在滚动条属性(scrollleft)发生改变时为元素添加平滑的过渡效果? 解决方案:scroll-behavior 属性 为滚动容器设置 scroll-behavior 属性可以实现平滑滚动。 html 代码: click the button to slide right!…

    2025年12月24日
    500
  • 如何选择元素个数不固定的指定类名子元素?

    灵活选择元素个数不固定的指定类名子元素 在网页布局中,有时需要选择特定类名的子元素,但这些元素的数量并不固定。例如,下面这段 html 代码中,activebar 和 item 元素的数量均不固定: *n *n 如果需要选择第一个 item元素,可以使用 css 选择器 :nth-child()。该…

    2025年12月24日
    200
  • 使用 SVG 如何实现自定义宽度、间距和半径的虚线边框?

    使用 svg 实现自定义虚线边框 如何实现一个具有自定义宽度、间距和半径的虚线边框是一个常见的前端开发问题。传统的解决方案通常涉及使用 border-image 引入切片图片,但是这种方法存在引入外部资源、性能低下的缺点。 为了避免上述问题,可以使用 svg(可缩放矢量图形)来创建纯代码实现。一种方…

    2025年12月24日
    100
  • 如何让“元素跟随文本高度,而不是撑高父容器?

    如何让 元素跟随文本高度,而不是撑高父容器 在页面布局中,经常遇到父容器高度被子元素撑开的问题。在图例所示的案例中,父容器被较高的图片撑开,而文本的高度没有被考虑。本问答将提供纯css解决方案,让图片跟随文本高度,确保父容器的高度不会被图片影响。 解决方法 为了解决这个问题,需要将图片从文档流中脱离…

    2025年12月24日
    000
  • 为什么 CSS mask 属性未请求指定图片?

    解决 css mask 属性未请求图片的问题 在使用 css mask 属性时,指定了图片地址,但网络面板显示未请求获取该图片,这可能是由于浏览器兼容性问题造成的。 问题 如下代码所示: 立即学习“前端免费学习笔记(深入)”; icon [data-icon=”cloud”] { –icon-cl…

    2025年12月24日
    200
  • 如何利用 CSS 选中激活标签并影响相邻元素的样式?

    如何利用 css 选中激活标签并影响相邻元素? 为了实现激活标签影响相邻元素的样式需求,可以通过 :has 选择器来实现。以下是如何具体操作: 对于激活标签相邻后的元素,可以在 css 中使用以下代码进行设置: li:has(+li.active) { border-radius: 0 0 10px…

    2025年12月24日
    100
  • 如何模拟Windows 10 设置界面中的鼠标悬浮放大效果?

    win10设置界面的鼠标移动显示周边的样式(探照灯效果)的实现方式 在windows设置界面的鼠标悬浮效果中,光标周围会显示一个放大区域。在前端开发中,可以通过多种方式实现类似的效果。 使用css 使用css的transform和box-shadow属性。通过将transform: scale(1.…

    2025年12月24日
    200
  • 为什么我的 Safari 自定义样式表在百度页面上失效了?

    为什么在 Safari 中自定义样式表未能正常工作? 在 Safari 的偏好设置中设置自定义样式表后,您对其进行测试却发现效果不同。在您自己的网页中,样式有效,而在百度页面中却失效。 造成这种情况的原因是,第一个访问的项目使用了文件协议,可以访问本地目录中的图片文件。而第二个访问的百度使用了 ht…

    2025年12月24日
    000
  • 如何用前端实现 Windows 10 设置界面的鼠标移动探照灯效果?

    如何在前端实现 Windows 10 设置界面中的鼠标移动探照灯效果 想要在前端开发中实现 Windows 10 设置界面中类似的鼠标移动探照灯效果,可以通过以下途径: CSS 解决方案 DEMO 1: Windows 10 网格悬停效果:https://codepen.io/tr4553r7/pe…

    2025年12月24日
    000
  • 使用CSS mask属性指定图片URL时,为什么浏览器无法加载图片?

    css mask属性未能加载图片的解决方法 使用css mask属性指定图片url时,如示例中所示: mask: url(“https://api.iconify.design/mdi:apple-icloud.svg”) center / contain no-repeat; 但是,在网络面板中却…

    2025年12月24日
    000
  • 如何用CSS Paint API为网页元素添加时尚的斑马线边框?

    为元素添加时尚的斑马线边框 在网页设计中,有时我们需要添加时尚的边框来提升元素的视觉效果。其中,斑马线边框是一种既醒目又别致的设计元素。 实现斜向斑马线边框 要实现斜向斑马线间隔圆环,我们可以使用css paint api。该api提供了强大的功能,可以让我们在元素上绘制复杂的图形。 立即学习“前端…

    2025年12月24日
    000
  • 图片如何不撑高父容器?

    如何让图片不撑高父容器? 当父容器包含不同高度的子元素时,父容器的高度通常会被最高元素撑开。如果你希望父容器的高度由文本内容撑开,避免图片对其产生影响,可以通过以下 css 解决方法: 绝对定位元素: .child-image { position: absolute; top: 0; left: …

    2025年12月24日
    000
  • CSS 帮助

    我正在尝试将文本附加到棕色框的左侧。我不能。我不知道代码有什么问题。请帮助我。 css .hero { position: relative; bottom: 80px; display: flex; justify-content: left; align-items: start; color:…

    2025年12月24日 好文分享
    200
  • 前端代码辅助工具:如何选择最可靠的AI工具?

    前端代码辅助工具:可靠性探讨 对于前端工程师来说,在HTML、CSS和JavaScript开发中借助AI工具是司空见惯的事情。然而,并非所有工具都能提供同等的可靠性。 个性化需求 关于哪个AI工具最可靠,这个问题没有一刀切的答案。每个人的使用习惯和项目需求各不相同。以下是一些影响选择的重要因素: 立…

    2025年12月24日
    300
  • 如何用 CSS Paint API 实现倾斜的斑马线间隔圆环?

    实现斑马线边框样式:探究 css paint api 本文将探究如何使用 css paint api 实现倾斜的斑马线间隔圆环。 问题: 给定一个有多个圆圈组成的斑马线图案,如何使用 css 实现倾斜的斑马线间隔圆环? 答案: 立即学习“前端免费学习笔记(深入)”; 使用 css paint api…

    2025年12月24日
    000
  • 如何使用CSS Paint API实现倾斜斑马线间隔圆环边框?

    css实现斑马线边框样式 想定制一个带有倾斜斑马线间隔圆环的边框?现在使用css paint api,定制任何样式都轻而易举。 css paint api 这是一个新的css特性,允许开发人员创建自定义形状和图案,其中包括斑马线样式。 立即学习“前端免费学习笔记(深入)”; 实现倾斜斑马线间隔圆环 …

    2025年12月24日
    100

发表回复

登录后才能评论
关注微信