php多进程编程时如何共享变量?
我写了个socket server,需要能共享父进程的$gsocks给子进程,但好像不能做到,我应该怎么做才能共享变量?
补充,额,memcache或者redis存的是字符串之类的把,我如何共享我问题代码中的socket res句柄?
代码如下:
服务端的:
<?php
$gsocks = array();
$pid = pcntl_fork();
switch ($pid) {
case -1:
#fail
break;
case 0:
for(;;){
sleep(1);
echo "C";
var_dump($gsocks);### 这里一直输出空数组,那么应该如何共享内存呢?
if (count($gsocks) == 0) continue;
$gsock = $gsocks[rand(0,count($gsocks)-1)];
$msg = time()." You are a lucky man \n";
socket_write($gsock, $msg, strlen($msg));
}
#i am child
break;
default:
//确保在连接客户端时不会超时
set_time_limit(0);
//设置IP和端口号
$address = "127.0.0.1";
$port = 2046; //调试的时候,可以多换端口来测试程序!
$sock = socket_create(AF_INET, SOCK_STREAM, SOL_TCP) or die("socket_create() 失败的原因是:" . socket_strerror(socket_last_error()) . "/n");
//阻塞模式
socket_set_block($sock) or die("socket_set_block() 失败的原因是:" . socket_strerror(socket_last_error()) . "/n");
//绑定到socket端口
$result = socket_bind($sock, $address, $port) or die("socket_bind() 失败的原因是:" . socket_strerror(socket_last_error()) . "/n");
//开始监听
$result = socket_listen($sock, 4) or die("socket_listen() 失败的原因是:" . socket_strerror(socket_last_error()) . "/n");
do { // never stop the daemon
//它接收连接请求并调用一个子连接Socket来处理客户端和服务器间的信息
$msgsock = socket_accept($sock) or die("socket_accept() failed: reason: " . socket_strerror(socket_last_error()) . "/n");
//读取客户端数据
//socket_read函数会一直读取客户端数据,直到遇见\n,\t或者\0字符.PHP脚本把这写字符看做是输入的结束符.
$buf = socket_read($msgsock, 8192);
// echo "Received msg: $buf \n";
//数据传送 向客户端写入返回结果
$msg = "welcome \n";
socket_write($msgsock, $msg, strlen($msg)) or die("socket_write() failed: reason: " . socket_strerror(socket_last_error()) ."/n");
//一旦输出被返回到客户端,父/子socket都应通过socket_close($msgsock)函数来终止
$gsocks[] = $msgsock;
if ($buf == "stop") {
break;
}
} while (true);
foreach ($gsocks as $gsock) {
socket_close($gsock);
}
socket_close($sock);
#parent
}
客户端的:
<?php
/**
* File name:client.php
* 客户端代码
*
* @author guisu.huang
* @since 2012-04-11
*/
set_time_limit(0);
$host = "127.0.0.1";
$port = 2046;
$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP)or die("Could not create socket\n"); // 创建一个Socket
$connection = socket_connect($socket, $host, $port) or die("Could not connet server\n"); // 连接
socket_write($socket, "hello socket") or die("Write failed\n"); // 数据传送 向服务器发送消息
while ($buff = socket_read($socket, 1024, PHP_NORMAL_READ)) {
echo("Response was:" . $buff . "\n");
}
socket_close($socket);
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(3)
用
memcache
或redis
之流即可搞定。共享变量在fork产生新进程的时候会失效的,但是变量名会传过去,因为它在全局符号表里面保存的。
同看到多进程这里,这种情况只需要将原先放在父进程中的
socket句柄
的创建逻辑放到pcntl_fork()
之前就行了,而不是fork后再在父进程中产生,这样父子进程都可以调用socket句柄
了,因为fork后父子间的变量就独立了,而fork前的变量父子都可以用。如果非要fork后再同步的话,可以搜索关键字
进程间传递socket描述符
等关键字,大体看了下,linux下貌似需要两个进程间再建立一个uninx domain socket
,然后用sendmsg
等方法,php方面socket_sendmsg
等函数底层是调用了此方法,但文档中连参数都没有介绍,需要自己去查找相关的用法。