在使用appnet时,如果只开一个worker进程可以使用静态变量来存储全局数据,但若在开启多个woker进程的情况下,静态变量作用域是在当前进程中,每个进程只能读写当前进程的数据,进程与进程间的数据是隔离的。就无法实现共享。这种情况下可以使用APCU扩展来实现进程间数据共享。
apcu前身是apc,大家知道apc缓存分为系统缓存和用户缓存,他们的区别是什么呢?
1、系统缓存是指php执行时增加缓存,减少php文件的反复检查和编译,从而达到系统加速的效果。
2、用户缓存是指,php代码中将数据写入缓存,是用户写入的数据,通过key和value的键值方式插入和读取。这种数据叫做用户缓存。
php5.5以后,opcache将代替apc做为php加速的位置,也就是代替其系统缓存的位置。并将用户缓存功能独立出来,开启新的组件,这个组件名称叫做apcu。
apcu下载地址:http://pecl.php.net/package/APCu
php.ini配置:
apc.enabled=1
apc.shm_size=32M
apc.enable_cli=1
以appnet webchat为例来介绍apcu的用法:
define( "WORKER_NUM" , 1 );
private static function setConnlist( $fd , $conn )
{
if( WORKER_NUM > 1 )
{
$list = apcu_fetch( "webchat_conn_list" );
$list[$fd] = $conn;
apcu_store( "webchat_conn_list" , $list );
}
else
{
self::$connections[$fd] = $conn;
}
}
private static function getConnlist()
{
if( WORKER_NUM > 1 )
{
$list = apcu_fetch( "webchat_conn_list" );
return $list;
}
return self::$connections;
}
这是webchat中设置或获取连接列表的方法,当用户在聊天室中发送一条消息时,当前用户所在的worker进程需要向所有进程的连接广播消息。所以需要获取全局的连接列表。
程序中对worker数量进行了判断,如果是单进程只需要使用静态变量self::$connections即可。
完整示例见:https://github.com/lchb369/appnet_php7/blob/master/example/server.php
其它可选方案:
redis/memcache:如果要做分布式存储可以使用,否则不推荐,因为redis/memcache需要tcp通信,即便是本地也需要unix domain socket通信,其效率远不如其于共享内存的apcu