$pid = pcntl_fork(); if ($pid == -1) { throw new Exception('fork子进程失败!'); } elseif ($pid == 0) { //子进程 $id = getmypid(); $data['message']='ceshi4'; $data['sub']='-1'; for($i=0;$i<10;$i++){ M('test')->add($data); } /* Close the connection 关闭连接*/ mysqli_close($link); posix_kill($id, 9); exit(0); }else{ //父进程 //sleep(10); }
这个是在tp里面的代码,如果将父进程的sleep(10)注释取消,子进程的数据插入便可以实现完成,并且在linux,通过命令ps -ef | grep httpd 可以看到一个已经kill掉的进程。如果将sleep(10)注释掉的话,那么子进程就会不执行,通过命令ps -ef | grep httpd看到的是一个没有kill掉的进程,依然存在,所以就是在插入mysql的时候出错了。为何会这样呢?
原因:TP连接数据库是使用的单例模式,在生成子进程后,两个进程共用一个单例子模式,由于需要要达到非阻塞异步的效果,主进程需要提前结束向客户端响应,随后子进程跑数据完成重任务。所以在主进程结束后便把Mysql连接资源给释放掉了,子进程自然不能使用,任务自然完成不了。
注意事项:
1. 在子进程中的变量,使用完之后,务必记得unset()注销变量,否则造成内存溢出
2. 子进程执行完毕之后,需要exit(0)退出程序,否则子进程无法退出,占用系统资源。
$pid = pcntl_fork(); if ($pid == -1) { throw new Exception('fork子进程失败!'); } elseif ($pid == 0) { //子进程 $id = getmypid(); $link = mysql_connect( '******', /* The host to connect to 连接MySQL地址 */ '***', /* The user to connect as 连接MySQL用户名 */ '****', /* The password to use 连接MySQL密码 */ true); /* The default database to query 连接数据库名称*/ mysql_select_db("youbibuluo", $link); $sql="INSERT INTO Persons (messsage, sub) VALUES ('child4','-1')"; for($i=0;$i<10;$i++){ mysql_query($sql,$link); } /* Close the connection 关闭连接*/ mysqli_close($link); posix_kill($id, 9); exit(0); }else{ //父进程 }
这样做便可以解决这个问题,子进程单独创造一个数据库链接,可以完成任务。但是在实际中,但凡使用导多进程的,肯定有大量的任务需要处理,而且很多已经成型的内部模块需要调用,所以直接原生的写会有很多问题,仅仅证明了上面的设想。接着研究在单列模式的数据库与多进程矛盾的TP框架中寻找更好的解决办法,或者自己独立写一个多进程驱动。目前这样设想。