PHP APCu缓存使用与避坑

  • 极简概括: PHP 的开源内存缓存扩展,类比Redis,但是一般都用Redis,所以APCu用的很少。

  • 官方文档:https://www.php.net/manual/zh/apcu.configuration.php

  • 解决问题:类比Redis做缓存组件,提升性能,同步数据使用。

  • 适用场景:轻量级的缓存,适合写少读多的场景。缺少原子性、缺少多条指令无间隙执行,不建议高并发时写多读多,写多读少的场景下使用。

  • 优点:

    • 比Redis快一百多倍。

    • 运维成本低:利用PHP扩展的方式实现,无需与缓存组件进行网络通信。

    • 简单易用:APCu 提供了简单而有效的接口,容易上手。

    • 跨文件跨进程:A文件set值,B文件get值,是可以获取到值的,若做不到和变量没区别。

  • 缺点:

    • 不支持远程独立部署。

    • 类型没有Redis的多,适用场景仅限于缓存。

    • 数据无法做到常驻内存,重启或出故障,数据丢了就没了,没有像Redis的RDB或AOF持久化机制。

    • 无法保证多个操作的原子性。

    • 可能存在超卖的问题。

    • 获取确定已经存在的值可能会遇到false,获取结果不稳定。

  • 注意:从PHP 8.0.0开始,不再支持apcu bc。

是否能像Redis+Lua一样保证多个操作原子性?

不能。
Redis是单线程的,意味着单位时间内只执行一个任务,Redis让并发过来的任务强制串行执行。有Lua的加持,保证多条指令无间隙执行。
APCu没有这个机制,让操作要么都成功,要么都失败,要实现这一步需要手动写逻辑。

高并发下会有超卖的一致性问题吗?

还好APCu有乐观锁机制,可以防止超卖问题。

但APCu 没有互斥的锁机制,互斥意味着并发过来的请求,通过独占该资源,让任务串行执行。

由于PHP+Nginx默认是多进程机制,(也可以调整为多线程,用得少)
假设一个场景:
获取到值,自增。进程P0获取到A的值为5的时候,想要自增到6,可能其它进程已经自增到8了。此时两步操作存在间隙,又没有机制对此数据加锁防止被其它进程更改,所以可能P0执行自增时会加到9,这是个概率问题,因此乐观锁的机制就显得非常重要。

安装

前提是安装好了PHP,默认在/usr/local/php下,并配置有/usr/local/php/bin目录的环境变量cd /test
wget https://pecl.php.net/get/apcu-5.1.23.tgz
tar zxf apcu-5.1.23.tgz
phpize
./configure
make
make install

相关配置(php.ini)

配置名值类型默认值说明
apc.enabledint1设置为0以禁用APC。这在APC被静态编译到PHP中时非常有用,因为没有其他方法可以禁用它。
apc.shm_segmentsint1为编译器缓存分配的共享内存段的数量。如果APC共享内存不足,但apc.shm_size设置为系统允许的最高值,提高该值可能会防止APC耗尽其内存。
apc.shm_sizeint32M每个共享内存段的大小
apc.entries_hintint4096是用于设置APCu缓存的条目预期数量
apc.ttlint0指定缓存中的条目在过期之前可以存在多长时间,单位为秒。默认为0,表示永不过期
apc.gc_ttlint3600指定过期缓存条目被清理的时间间隔,单位为秒
apc.mmap_file_maskstringnull是用于配置在使用共享内存映射(MMAP)方式时的文件名模板。这个选项在某些情况下可以用于解决操作系统限制或者提高性能。默认情况下,这个选项为空,APCu会使用系统默认的文件名模板。设置apc.mmap_file_mask时,你可以使用一些特殊的占位符来指定文件名的格式,例如%s代表共享内存标识符的十六进制表示,%p代表当前进程的PID(进程标识符)。这样可以确保每个进程使用不同的文件名,避免冲突。一般情况下,你不需要手动设置这个选项,除非你遇到共享内存映射方面的特定问题或者有特殊需求。在大多数情况下,使用默认设置即可满足需求
apc.slam_defenseint1防止缓存雪崩,多进程下,每个进程都试图同时缓存同一个文件。此选项设置跳过尝试缓存未缓存文件的进程的百分比。或者将其视为单个进程跳过缓存的概率。例如,设置为75意味着该进程有75%的概率不会缓存未缓存的文件。因此,设置越高,对缓存雪崩的防御就越强。将此项设置为0禁用此功能
apc.enable_cliint0是否在cli模式下启用apc,实测不生效
apc.use_request_timeint0置控制是否APC应该使用请求时间来为文件加上时间戳。当启用时,它可以确保在请求时间变化时刷新缓存文件,这在某些情况下会很有用,比如在开发或调试代码时
apc.serializerstringphp用于配置APC序列化方式。
apc.coredump_unmapint0启用APC处理信号,如SIGSEGV,该信号在收到信号时写入内核文件。当收到这些信号时,APC将尝试取消共享内存段的映射,以便将其从核心文件中排除。当接收到致命信号并且配置了大型APC共享内存段时,此设置可以提高系统稳定性
apc.preload_pathstringnull用于指定要预加载的PHP文件或目录的路径。预加载可以提高应用程序的性能,因为它可以在应用程序启动时将指定的文件或目录加载到内存中,从而减少了每次请求时的文件读取和解析时间。

使用

设置值,注意,缓存有值的情况下无法设置值,类比Redis的setnx,类型支持标量、数组、与对象,这一点非常好。bool  apcu_add(key, val, ttl);

获取缓存,获取不到返回false,并发情况下容易返回falsemixed apcu_fetch(key);

乐观锁机制,在旧值的基础上添加新的值bool apcu_cas(key, int_old, int_new):

清除所有缓存bool apcu_clear_cache()

递减,参数2支持负数int apcu_dec(key, 递减值, 函数返回结果赋值给变量, ttl秒)

从缓存中删除某个元素bool|array apcu_delete(array|string key)

判断当前环境能否使用apcubool apcu_enabled()

若key不存在,则调用callback,并带有一个默认参数,即key的值null apcu_entry(key, callback, ttl)

判断多个key或者单个key是否存在。当参数为array时,函数返回只存在的key组成的数组bool|array apcu_exists(string|array key)

获取某个key的值,若参数1是数组,那么结果也是个数组,只会返回存在的key的值,若key有值参数2为true,否则反之。bool|array apcu_fetch(array|string key, $var);

递增,参数2支持负数int apcu_inc(key, 递增值, 函数返回结果赋值给变量, ttl秒)

将key的值存储缓存,类比Redis的set,若已存在,可直接替换,参数1也可以传输数组。bool apcu_store(array|string key, val, ttl)

压测,对比连接Redis性能

方式轮次APCu耗时(秒)Redis耗时(秒)
只读100000.0111.162
只写100000.0121.062
读写,一次new Redis100000.0112.117
读写,多次new Redis100000.0113.646
只读(APCu):<?php$start = microtime(true);for($i = 0; $i < 10000; $i++) {    $key = 'apcu'. $i;    apcu_fetch($key);
}echo microtime(true) - $start;


只读(Redis):<?php$redis = new Redis();$redis->connect('127.0.0.1', 6379);$start = microtime(true);for($i = 0; $i < 10000; $i++) {    $key = 'redis' . $i;    $redis->get($key);
}echo microtime(true) - $start;


只写(APCu):<?php$start = microtime(true);for($i = 0; $i < 10000; $i++) {    $key = 'apcu'. $i;    apcu_add($key, $i);
}echo microtime(true) - $start;


只写(Redis):<?php$redis = new Redis();$redis->connect('127.0.0.1', 6379);$start = microtime(true);for($i = 0; $i < 10000; $i++) {    $key = 'redis' . $i;    $redis->set($key, $i);
}echo microtime(true) - $start;


读写,一次new Redis(APCu):<?php$start = microtime(true);for($i = 0; $i < 10000; $i++) {    $key = 'apcu'. $i;    apcu_add($key, $i);    apcu_fetch($key);
}echo microtime(true) - $start;


读写,一次new Redis(Redis):<?php$redis = new Redis();$redis->connect('127.0.0.1', 6379);$start = microtime(true);for($i = 0; $i < 10000; $i++) {    $key = 'redis' . $i;    $redis->set($key, $i);    $redis->get($key);
}echo microtime(true) - $start;


读写,多次new Redis(APCu):<?php$start = microtime(true);for($i = 0; $i < 10000; $i++) {    $key = 'apcu'. $i;    apcu_add($key, $i);    apcu_fetch($key);
}echo microtime(true) - $start;


读写,多次new Redis(Redis):<?php$start = microtime(true);for($i = 0; $i < 10000; $i++) {$redis = new Redis();$redis->connect('127.0.0.1', 6379);    $key = 'redis' . $i;    $redis->set($key, $i);    $redis->get($key);
}

高并发下对APCu原子性测试

压测工具用ApiPOST,我认为比ab工具好用。
压测前,为了保证ApiPOST压测参数(压测轮次 * 并发数 结果积)的准确性,特地用Redis做了多次测试,发现参数是对的,并发数大了就不对(150以上),这意味着压测工具应该没问题,只是设备线程数不够。

<?php$redis = new Redis();$redis->connect('127.0.0.1', 6379);$redis->decr('test_key');

多条APCu语句执行能才能测试更能充分原子性,并发100,测试10轮次,也就是1000次请求,但是多次压测下来,结果不对。apcu_fetch获取值动不动就是false,导致结果重新赋值为10000(在不存在的情况下赋初始值),有并发问题,但不是因为并发引起的,而是因为apcu_fetch函数的问题,获取不到值返回false。

<?php$key = 'test_key';$res = apcu_fetch($key);if($res === false) {    apcu_add($key, 10000);
} else {    apcu_delete($key);    apcu_add($key, $res - 1);
}echo apcu_fetch($key);

原文地址:https://www.cnblogs.com/phpphp/p/18200989

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


极简概括: PHP 的开源内存缓存扩展,类比Redis,但是一般都用Redis,所以APCu用的很少。官方文档:https://www.php.net/manual/zh/apcu.configuration.php解决问题:类比Redis做缓存组件,提升性能,同步数据使用。
请看如下代码: $list = [1,2,4,5]; $list2 = [5,6,7,9]; foreach ($list as $key =&gt; &amp;$value) { $value = strval($value); } foreach ($list2 as $key =&gt; $v
一、nginx 安装 1. 在nginx官网下载nginx源码&#x9;提供一个nginx官网下载地址: http://nginx.org/download/nginx-1.12.2.tar.gz 注意:请先确认 是否已经安装过 pcre pcre-devel openssl openssl-devel z
先看图 数据条数:9000+ 第1,2行,golangʾxcelize方式导出,耗时:5s 第3行,PHP+xlswriter方式导出,耗时:2min 一、介绍 xlswriter是一个高效处理excel文件的PHP扩展,底层以C语言实现;处理速度是PHPExcel几十倍甚至几百倍的效率。 官方链
今天使用Thinkphp5做异步任务传递where参数时遇到一个问题: 有一段如下代码: $where[&#39;jst.supplier&#39;] = [&#39;exp&#39;, Db::raw(&#39;&gt;0 or jst.is_supplier=1&#39;)]; 在使用swool
汇总 PHP5.1: autoload PDO MySQLi 类型约束 PHP5.2: JSON 支持 PHP5.3: 命名空间 匿名函数 闭包 新增魔术方法__callStatic()和__invoke() 新增魔术变量__DIR__ 动态调用静态方法 延迟静态绑定 Heredoc和 Nowdoc
文章浏览阅读8.4k次,点赞8次,收藏7次。SourceCodester Online Tours & Travels Management System pay.php sql injectionLine 16 of pay.php invokes a SQL query built using unvalidated input. This call could allow an attacker to modify the statement’s meaning or to execute arbitrary SQL commands.SQL
文章浏览阅读3.4k次,点赞46次,收藏51次。本文为大家介绍在windwos系统搭建typecho博客+cpolar内网穿透工具将博客发布到公共网络环境,实现远程也可以访问和操作。_windows搭建typecho
文章浏览阅读1.1k次。- php是最优秀, 最原生的模板语言, 替代语法,让php更加的优雅的与html生活在一起 -->请放心, 最终生成的,或者说用户最终看到的,仍然是一个html文档, php代码中的内容不会被泄漏的。-- 将php与html代码混编的时候,大括号很容易造成配对错误,最好杜绝它 -->php标签内部代码由php.exe解释, php标签之外的代码原样输出,仍由web服务器解析。-- 所以php的流程控制语句, 都提供了替代语法,用冒号代替大括号 -->php echo '百变鹏仔'?_利用php将静态页面修改为动态页面
文章浏览阅读1.1k次,点赞18次,收藏15次。整理K8s网络相关笔记博文内容涉及 Linux network namespace 认知以及彼此通信Demo,实际中的应用理解不足小伙伴帮忙指正不必太纠结于当下,也不必太忧虑未来,当你经历过一些事情的时候,眼前的风景已经和从前不一样了。——村上春树。_linux network namespace 多端通信 模式认知
文章浏览阅读1.2k次,点赞22次,收藏19次。此网络模型提供了一个逻辑二层(L2)网络,该网络封装在跨 Kubernetes 集群节点的现有三层(L3)网络拓扑上。使用此模型,可以为容器提供一个隔离的 L2 网络,而无需分发路由。封装网络带来了少量的处理开销以及由于覆盖封装生成 IP header 造成的 IP 包大小增加。封装信息由 Kubernetes worker 之间的 UDP 端口分发,交换如何访问 MAC 地址的网络控制平面信息。此类网络模型中常用的封装是 VXLAN、Internet 协议安全性 (IPSec) 和 IP-in-IP。_k8s网络组件对比
文章浏览阅读1.1k次,点赞14次,收藏19次。当我们谈论网络安全时,我们正在讨论的是保护我们的在线空间,这是我们所有人的共享责任。网络安全涉及保护我们的信息,防止被未经授权的人访问、披露、破坏或修改。
文章浏览阅读1.3w次,点赞3次,收藏7次。尽管您可以通过 ping 命令解析出网站的 IP 地址,但是可能在浏览器中访问时仍然遇到问题,这可能是因为浏览器使用的 DNS 解析结果不同于 ping 命令使用的解析结果。可能是因为您的网络或设备上设置了防火墙,阻止了对特定网站的访问。有些国家或组织可能会对特定的域名进行屏蔽,从而阻止访问相关网站。如果您的网络使用代理服务器进行访问控制,可能会由于代理服务器的配置问题导致无法访问某些网站。即使您的网络和设备一切正常,目标网站本身可能也存在问题,例如服务器故障、维护或过载,导致无法访问。_能ping通打不开网页
文章浏览阅读839次,点赞22次,收藏19次。本系统带文档lw万字以上文末可领取本课题的JAVA源码参考。
文章浏览阅读2.1k次,点赞31次,收藏22次。基于微信小程序奶茶点餐外卖系统设计与实现(PHP后台+Mysql)可行性分析毕设源代码毕业设计,数据安全和系统稳定性以及团队能力和资源配备方面都具备较好的条件。因此,该项目的可行性较高。:黄菊华老师《Vue.js入门与商城开发实战》《微信小程序商城开发》图书作者,CSDN博客专家,在线教育专家,CSDN钻石讲师;微信小程序作为一种快捷、方便的移动应用形式,成为很多用户点餐外卖的首选。项目的界面和功能都可以定制,包安装运行!项目配有对应开发文档、开题报告、任务书、PPT、论文模版等。
文章浏览阅读1.8k次,点赞52次,收藏38次。本文主要通过对系统的前台系统和后台管理系统进行了功能性需求分析,对系统的安全性和可扩展性进行了非功能性需求分析。在详细的需求分析的基础上,根据系统的功能设计确定了数据库结构,实现完整的代码编写。Lucky+Baby母婴用品网站使用 Dreamweaver、HBuilder代码编辑器、Apache服务器等开发工具,完成了系统的主要模块的页面设计和功能实现。本文展示了首页页面的实现效果图,并通过代码和页面介绍了用户注册功能、商品搜索功能、生成订单和查看我的订单功能、在线付款功能功能的实现过程。
文章浏览阅读1.5k次,点赞45次,收藏40次。本设计主要实现集人性化、高效率、便捷等优点于一身的人事信息管理系统,完成首页、系统用户、通知公告、部门信息、员工薪资、考勤签到、员工请假、招聘信息、应聘信息等功能模块。
文章浏览阅读1k次。该错误通常出现在数据库读取结果集数据时,比如当我们写好SQL语句从数据库读取数据时,本身应该返回结果集,再给结果集中读取数据。解决思路:这种错误一般是因为echo后面输出了一个数组导致的,或者是数组作为字符串进行拼接运算时导致的。该错误直译为:警告:mysqli_fetch_assoc函数期望参数1是mysqli的结果集,但是给了一个布尔值。这种错误是PHP解析器在解析时遇到了语法错误,直译为:解析错误:语法错误,意料之外的...该错误直译为:提示:未定义的索引:username。_array to string conversion in
文章浏览阅读2.7w次。解决http请求报错context deadline exceeded (Client.Timeout exceeded while awaiting headers)_context deadline exceeded (client.timeout exceeded while awaiting headers)
文章浏览阅读1.3k次,点赞26次,收藏24次。复杂网络是一种由大量相互连接的元素(节点或顶点)组成的网络结构,这些连接通常是非常复杂和动态的。这些网络可以在各种领域中发现,包括社交网络、生物学系统、信息技术和交通系统等。_代理建模