PHP进程通信-信号量和共享内存

本文深入探讨了进程间通信的高效方式——共享内存,并详细解释了如何通过信号量解决共享内存带来的数据同步问题。通过一个PHP示例代码,展示了如何使用ftok、sem_get、shm_attach等函数创建和管理共享内存及信号量。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

信号量与共享内存。共享内存是最快是进程间通信方式,因为n个进程之间并不需要数据复制,而是直接操控同一份数据。实际上信号量和共享内存是分不开的,要用也是搭配着用。*NIX的一些书籍中甚至不建议新手轻易使用这种进程间通信的方式,因为这是一种极易产生死锁的解决方案。共享内存顾名思义,就是一坨内存中的区域,可以让多个进程进行读写。这里最大的问题就在于数据同步的问题,比如一个在更改数据的时候,另一个进程不可以读,不然就会产生问题。所以为了解决这个问题才引入了信号量,信号量是一个计数器,是配合共享内存使用的.

<?php
//ftok将一个路径名pathname和一个项目名(必须为一个字符), 转化成一个整形的用来使用系统V IPC的key
$sem_key = ftok( __FILE__, 'b' );
// 获取信号id
$sem_id = sem_get( $sem_key );
//ftok将一个路径名pathname和一个项目名(必须为一个字符), 转化成一个整形的用来使用系统V IPC的key
$shm_key = ftok( __FILE__, 'm' );
// 创建一个共享内存
$shm_id = shm_attach( $shm_key, 1024, 0666 );
const SHM_VAR = 1;
$child_pid = [];
for( $i = 1; $i <= 2; $i++ ){
    $pid = pcntl_fork();
    if( $pid < 0 ){
        exit();
    } else if( 0 == $pid ) {
        // 获取信号量(获取锁)
        sem_acquire( $sem_id );
        // 判断此信号量中是否有值
        if( shm_has_var( $shm_id, SHM_VAR ) ){
            $counter = shm_get_var( $shm_id, SHM_VAR );
            $counter += 1;
            shm_put_var( $shm_id, SHM_VAR, $counter );
        } else {
            $counter = 1;
            shm_put_var( $shm_id, SHM_VAR, $counter );
        }
        // 释放信号量(释放锁)
        sem_release( $sem_id );
        exit;
    } else if( $pid > 0 ) {
        $child_pid[] = $pid;
    }
}
while( !empty( $child_pid ) ){
    foreach( $child_pid as $pid_key => $pid_item ){
        pcntl_waitpid( $pid_item, $status, WNOHANG );
        unset( $child_pid[ $pid_key ] );
    }
}
sleep( 2 );
echo '最终结果'.shm_get_var( $shm_id, SHM_VAR ).PHP_EOL;
// 删除共享内存数据,删除共享内存是有顺序的,先remove后detach
shm_remove( $shm_id );
shm_detach( $shm_id );
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值