php进程通信之信号量的用途是什么

常见进程通信方式

一些理论基础

  • 临界资源:每次仅允许一个进程访问的资源。

  • 临界区:每个进程中访问临界资源的那段代码叫临界区

所谓临界区(也称为临界段)就是访问和操作共享数据的代码段。

进程互斥:两个或以上的进程不能同时进入关于同一组共享变量的临界区域,即一个进程正在访问临界资源,另一个进程要想访问必须等待。

进程同步:主要研究如何确定数个进程之间的执行顺序和避免数据竞争的问题 即,如何让多个进程能一块很好的协作运行

所谓同步,就是并发进程/线程在一些关键点上可能需要互相等待与互通消息,这种相互制约的等待与互通信息称为进程/线程同步。

举个生活的同步例子,你肚子饿了想要吃饭,你叫妈妈早点做菜,妈妈听到后就开始做菜,但是在妈妈没有做完饭之前,你必须阻塞等待,等妈妈做完饭后,自然会通知你,接着你吃饭的事情就可以进行了。

注意,同步与互斥是两种不同的概念:

同步就好比:「操作 A 应在操作 B 之前执行」,「操作 C 必须在操作 A 和操作 B 都完成之后才能执行」等;

互斥就好比:「操作 A 和操作 B 不能在同一时刻执行」;

system V 信号量

信号量用途:主要用于多进程或多线程对公共资源对象的访问控制。 用来解决多进程(多线程同步的问题),类似于一把锁,访问前获取锁(获取不到则等待),访问后释放锁。

多进程/多线程一般是并发执行,如果对公共资源访问没有做同步处理,很容易造成数据破坏

信号量其实是一个整型的计数器,主要用于实现进程间的互斥与同步,而不是用于缓存进程间通信的数据。

信号量表示资源的数量,控制信号量的方式有两种原子操作:

一个是 P 操作,这个操作会把信号量减去 -1,相减后如果信号量 < 0,则表明资源已被占用,进程需阻塞等待;相减后如果信号量 >= 0,则表明还有资源可使用,进程可正常继续执行。

另一个是 V 操作,这个操作会把信号量加上 1,相加后如果信号量 <= 0,则表明当前有阻塞中的进程,于是会将该进程唤醒运行;相加后如果信号量 > 0,则表明当前没有阻塞中的进程;

P 操作是用在进入共享资源之前,V 操作是用在离开共享资源之后,这两个操作是必须成对出现的。

举个类比,2 个资源的信号量,相当于 2 条火车轨道,PV 操作如下图过程:

php进程通信之信号量的用途是什么

一辆火车进入轨道,相当于信号量的P操作,资源-1,这样就剩下一条轨道

php进程通信之信号量的用途是什么

接着又一辆火车占用另一条轨道,也就是P操作,资源-1

php进程通信之信号量的用途是什么

此时交通信号灯变为红色,因为没有轨道可用,第三辆火车必须等待

php进程通信之信号量的用途是什么

名列前茅辆火车离开轨道,相当于V操作,此时轨道资源为1,交通灯变为绿灯

php进程通信之信号量的用途是什么

第三辆火车发现交通信号灯变绿,于是进入火车轨道,轨道资源耗尽为0,于是交通信号灯变为红灯

php进程通信之信号量的用途是什么

在这个火车轨道系统中,轨道是公共资源,每辆火车好比一个线程,交通信号灯起的就是信号量的作用。信号量可以实现锁的互斥操作,也可以实现进程/线程同步

信号量类型

1)二进制信号量(也叫二值信号量)

此时信号量的初值只能是0和1。(二进制信号量可以实现互斥锁操作)

2)一般/计数信号量

此时信号量的初值可以是任意非负数。显然,其包含二进制信号量。上面举的火车轨道例子就可以使用计数信号量来实现,一般计数信号量与锁的区别是它可以允许多个线程/进程(线程的数量由计数信号量初值定义) 同时操作公共资源

一般只有在开发多进程的时候才可能遇到需要使用信号量的场景,phper 几乎很少有使用信号量的场景,就算有多进程对公共资源操作,大多也是使用 flock 文件锁做互斥操作

php模拟多进程操作公共资源

<?php$file = "num.txt";//定一个空文件$count =0;file_put_contents($file,$count);$pid = pcntl_fork();//fork 一个进程if($pid == 0){//子进程执行逻辑    $x = (int)file_get_contents($file);//读取文件内容    //i 循环累加    for($i=0; $i<1000; $i++){        $x = $x + 1;    }    //写入文件    file_put_contents($file,$x);    //子进程退出    exit(0);}//父进程执行逻辑$x = (int)file_get_contents($file);for($i=0; $i<1000; $i++){    $x = $x+1;}//累加写入file_put_contents($file,$x);

在编写一个shell 脚本辅助

#!/bin/bashfor a in {1..1000}do    (php demo1.php)    b=`cat num.txt`    if [ $b != 2000 ]    then        echo -e "错误$b"    fidone

按理来说,变量 $x 最后写入文件的值应该是2000,但很不幸,并不是如此,我们对上面的脚本执行一下:

php进程通信之信号量的用途是什么

运行了1000次,发现出现了变量$x值结果是 1000 的有8次,虽然发生错误的概率比较小,但是在计算机里是不能容忍的。

为什么会出现这种情况,我们知道单核cpu系统里为了实现多个程序同时运行的假象,操作系统通常都采用时间片调度,一个进程时间片用完就切换下一个进程运行,加上我们的高级语言不是每一行代码都是原子性的,比如x = (int)file_get_contents($file) 这行代码对于我们来说是不可分割是原子性的,但是经过编译器编译成汇编码【机器指令】可能是多条指令实现,这样就会出现问题,如果指令只执行到一半进程分配的时间片用完或者被其他进程打断,都有可能造成数据损坏,导致最后计算结果出现误差

使用php封装system v 信号量集函数

<?php$file = "num.txt";//定一个空文件$count =0;$key = ftok("demo1.php","x");$sem_id = sem_get($key,1);// 第二个参数是个整数,表示设置信号量集,设置为1 把它当做二值信号量来用,用于互斥file_put_contents($file,$count);$pid = pcntl_fork();//fork 一个进程if($pid == 0){//子进程执行逻辑sem_acquire($sem_id); // P -1 操作 获取一个信号量 , 如果为0表示资源被占用进程挂起等待信号量释放    $x = (int)file_get_contents($file);//读取文件内容    //i 循环累加    for($i=0; $i<1000; $i++){        $x = $x + 1;    }    //写入文件    file_put_contents($file,$x);       sem_release($sem_id); //V +1 操作 释放信号量    //子进程退出    exit(0);}//父进程执行逻辑sem_acquire($sem_id); // P -1 操作  获取信号量, 如果为0表示资源被占用进程挂起等待信号量释放$x = (int)file_get_contents($file);for($i=0; $i<1000; $i++){    $x = $x+1;}//累加写入file_put_contents($file,$x);sem_release($sem_id); //V +1 操作 释放信号量

加入信号量后,那就一定保证100%是2000,绝对不会出现其他数值。

感谢各位的阅读,以上就是“php进程通信之信号量的用途是什么”的内容了,经过本文的学习后,相信大家对php进程通信之信号量的用途是什么这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是亿速云,小编将为大家推送更多相关知识点的文章,欢迎关注!

文章标题:php进程通信之信号量的用途是什么,发布者:亿速云,转载请注明出处:https://worktile.com/kb/p/22479

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022年9月2日 下午4:40
下一篇 2022年9月2日

相关推荐

  • Mysql中锁的使用场景是什么

    Mysql中锁的使用场景主要有:常见锁类型、Mysql引擎介绍、常用引擎间的区别、共享锁与排他锁、排他锁的实际应用、共享锁的实际应用、死锁的发生、另一种发生死锁的情景、死锁的解决方式、意向锁和计划锁、乐观锁和悲观锁。 一、常见锁类型 表级锁,锁定整张表 页级锁,锁定一页 行级锁,锁定一行 共享锁,也…

    2022年6月29日
    26600
  • 【缺陷周话】第31期:错误的内存释放

    1、错误的内存释放方法 C语言中常见的内存申请函数包括malloc()、 realloc()、 calloc(),它们虽然功能不同,但都对应同一个内存释放函数 free(),C++中对内存的申请和释放采用new/delete、new []/delete[] 方式。不管是 C 语言还是 C++ 语言,…

    2022年9月20日
    12600
  • word字体放大后上半部分缺失怎么解决

    解决方法 1、首先我们把缺失的文字框选出来。 2、鼠标右键点击之后选择段落。 3、然后选择行距,这里点击单倍行距。然后点击确定。 4、这样就能看见字体恢复了正常。 读到这里,这篇“word字体放大后上半部分缺失怎么解决”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,…

    2022年9月11日
    23500
  • mysql怎么增加少数索引

    三种增加方法:1、使用“CREATE INDEX”语句增加,语法“CREATE UNIQUE INDEX 索引名 ON 表名(列名列表);”;2、建表时增加,语法“CREATE TABLE 表名(… UNIQUE KEY(列名列表) );”;3、修改表时增加,语法“ALTER TABLE…

    2022年9月22日
    2.6K00
  • web中常用的加密算法有哪些

    加密算法我们整体可以分为:可逆加密和不可逆加密,可逆加密又可以分为:对称加密和非对称加密。 一、不可逆加密 常见的不可逆加密算法有MD5,HMAC,SHA1、SHA-224、SHA-256、SHA-384,和SHA-512,其中SHA-224、SHA-256、SHA-384,和SHA-512我们可以…

    2022年8月29日
    43200
  • Nmap是如何识别主机指纹的

    0x01 Nmap维护一个nmap-os-db数据库,存储了上千种操作系统信息,简单一点来说,Nmap通过TCP/IP协议栈的指纹信息来识别目标主机的操作系统信息,这主要是利用了RFC标准中,没有强制规范了TCP/IP的某些实现,于是不同的系统中TCP/IP的实现方案可能都有其特定的方式,这些细节上…

    2022年9月21日
    37600
  • easyrecovery怎么恢复视频

    easyrecovery恢复视频的方法 1、选择恢复视频,点击下一步。 2、接着选择你需要恢复的硬盘进行扫描。 3、等待软件扫描完成。 4、经过扫描后可以直观的看到文件数,文件夹数,和可恢复数据数 5、找到你需要恢复的视频,右键恢复就可以了。 关于“easyrecovery怎么恢复视频”这篇文章的内…

    2022年9月26日
    11100
  • 怎么搭建配置Docker私有仓库

    ⛳️ 1.Docker容器三要素 Docker 是一个开源的应用容器引擎,基于 Go 语言 并遵从 Apache2.0 协议开源。Docker 可以让开发者打包他们的应用以及依赖包到一个轻量级、可移植的容器中,然后发布到任何流行的 Linux 机器上,也可以实现虚拟化。容器是完全使用沙箱机制,相互之…

    2022年9月18日
    36600
  • Javascript如何实现没有map()的映射数组

    没有map()的映射数组 我们先介绍一下map方法。map()方法返回一个新数组,数组中的元素为原始数组元素调用函数处理后的值,它会按照原始数组元素顺序依次处理元素。注意: map()不会改变原始数组,也不会对空数组进行检测。 下面我们来实现一个没有map的数组映射: // array.map(fu…

    2022年8月31日
    13000
  • 如何进行bee-box LDAP注入的靶场练习

    如果说sql注入的本质是拼接字符串的话,那么一切可以注入的本质都是拼接字符串,LDAP注入作为注入的一种也不例外,更有趣一点的说它是在拼接圆括号(sql注入也拼接圆括号,但是更习惯性的是说它拼接字符串)。 在环境配置篇里面已经很详细的说了bee-box中ldap环境的配置,靶场练习篇更多的是php与…

    2022年9月18日
    29700
联系我们
站长微信
站长微信
电话联系

400-800-1024

工作日9:30-21:00在线

分享本页
返回顶部