摘要:如果是这样,就需要引入信号量进行控制。这应该是因为正在共享内存被上一次操作占用中还没有释放导致。
共享内存
共享内存的使用主要是为了能够在同一台机器不同的进程中共享一些数据,比如在多个 php-fpm 进程中共享当前进程的使用情况。这种通信也称为进程间通信(Inter-Process Communication),简称 IPC。
PHP 内置的 shmop 扩展 (Shared Memory Operations) 提供了一系列共享内存操作的函数(可能是用的人不多吧,这一块儿的文档还没有中文翻译)。在 Linux 上,这些函数直接是通过调用 shm* 系列的函数实现,而 Winodows 上也通过对系统函数的封装实现了同样的调用。
主要函数:
shmop_close — 关闭共享内存块
shmop_delete — 删除共享内存块
shmop_open — 创建或打开共享内存块
shmop_read — 从共享内存块中读取数据
shmop_size — 获取共享内存块的大小
shmop_write — 向共享内存块中写入数据
与此相关的还有一个很重要的函数:ftok,通过文件的 inode 信息(*nix 上通过 stat 或 ls -i 命令查看)创建 IPC 的唯一 key(文件/文件夹的 inode 是唯一的)。这个函数在 Linux 上也是直接调用同名的系统函数实现,Windows 上还是使用一些封装。
一个简单的计数例子:
以上这段代码没执行一次计数加 1,而且数据是在不同进程之间共享的。也就是说除非手动删除这块内存使用,否则这个数据是不会重置的。
有个需要稍微注意的点:shmop_open 的第二个参数是个 flag,类似 fopen 的第二个参数,其取值有以前几个:
"a" 只读访问;
"c" 如果内存片段不存在,则创建,如果存在,则可读写;
"w" 读写;
"n" 创建新的内存片段,如果同样 key 的已存在,则会创建失败,这是为了安全使用共享内存考虑。
此外,由于使用的共享内存片段是固定长度的,在存储和读取的时候要计算好数据的长度,不然可能会写入失败或者读取空值。
信号控制既然上面使用到了共享内存存储数据,就需要考虑是否有多个进程同时写入数据到共享内存的情况,是否需要避免冲突。如果是这样,就需要引入信号量进行控制。
PHP 也提供了类似的内置扩展 sysvsem(这个扩展在 Windows 环境下没有,文档中将 ftok 函数也归到这个扩展中,但实际上 ftok 是在标准函数库中提供的,所以在 Windows 下也是可用的)。
在说信号量控制之前,先说另外一件有意思的事情:看官方文档你会发现这里同样也有共享内存操作的函数(shm_*),因为这其实是同一类别(或者说来自于同一作者)的三个扩展,还有一个是 sysvmsg(队列消息) 。函数的实现上稍有差别,但实际做的事情基本相同。这和上文的 shmop 扩展有什么区别呢?shmop 源码下的 README 文件有简单的说明:
PHP already had a shared memory extension (sysvshm) written by Christian Cartus
, unfortunately this extension was designed with PHP only in mind and offers high level features which are extremely bothersome for basic SHM we had in mind. 简单说来:sysvshm 扩展提供的方法并不是原封不动的存储用户的数据,而是先使用 PHP 的变量序列化函数对参数进行序列化然后再进行存储。这就导致通过这些方法存储的数据无法和非 PHP 进程共享。不过这样也能存储更丰富的 PHP 数据类型,上文的扩展中 shmop_write 只能写入字符串。那么为什么 sysvshm 同样不支持 Windows 呢?因为其并没有引入封装了 shm* 系列函数的 tsrm_win32.h 的头文件。
引入信号控制之后的示例:
但是本地想模拟实现写入冲突实际上是非常难的(考虑到计算机的执行速度)。在本地测试中,使用 for 循环操作时如果不使用 shmop_close 关闭资源会出现无法打开共享内存的错误警告。这应该是因为正在共享内存被上一次操作占用中还没有释放导致。
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/30350.html
摘要:本人前一遍中提到,当进程数过多时,多进程插入数据库表,超过了最大连接数,就会报错,插入就会有失败的情况。这样就可以控制有的连接数不会达到上限,程序不会出现连接数据库失败的情况。多进程耗内存比较严重,下一步打算用多线程来试试。 本人前一遍blog中提到,当进程数过多时,多进程插入mysql数据库表,超过了mysql最大连接数,就会报错,插入就会有失败的情况。想通过进程间通信来控制一下连接...
摘要:接受不到消息消息队列通过指定而被创建后,任意一方销毁了该队列,都会导致其他发送或接收方失败。用法场景进程,中代码段要用到中代码段的结果。完成了进程间同步问题此外进程间通信采用的方式是共享内存。 参考文章 深刻理解Linux进程间通信(IPC) 进程间通信(IPC)介绍 php高级应用之进程控制及进程间通讯 workman 作者发布 PHP 相关进程间通信扩展 -- System V ...
摘要:目的综上所述,我的目标就是实现基于模式实现的多进程管理工具。备注下文中,父进程统称为子进程统称为。最后我们通过下图来简单的总结和描述这个多进程实现的过程控制上面实现了多进程和多进程的常驻内存,那如何去管理呢答案多进程通信。 _ | | _ __ __ _ _ __...
摘要:为了解决这个问题,我们必须引入互斥机制。实现互斥机制的最简单办法就是使用信号灯。信号量是另外一种进程间的方式,它同其他机构管道消息队列不同。在这个停车场系统中,车位是公共资源,每辆车好比一个线程,看门人起的就是信号量的作用。 在单独的一个PHP进程中读写、创建、删除共享内存方面上你应该没有问题了。但是实际运行中不可能只是一个PHP进程在运行中。如果在多个进程的情况下你还是沿用单个进程的...
阅读 513·2023-04-25 14:26
阅读 1248·2021-11-25 09:43
阅读 3440·2021-09-22 15:25
阅读 1409·2019-08-30 15:54
阅读 478·2019-08-30 12:57
阅读 718·2019-08-29 17:24
阅读 3128·2019-08-28 18:13
阅读 2617·2019-08-28 17:52