在使用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