您好,欢迎访问一九零五行业门户网

PHP异步非阻塞实现方法

为让 php 在后端处理长时间任务时不阻塞,快速响应页面请求,可以有如下措施:
1.使用 fastcgi_finish_request()
如果 php 与 web 服务器使用了 php-fpm(fastcgi 进程管理器),那通过 fastcgi_finish_request() 函数能马上结束会话,而 php 线程可以继续在后台运行。
echo "program start...";file_put_contents('log.txt','start-time:'.date('y-m-d h:i:s'), file_append);fastcgi_finish_request();sleep(1);echo 'debug...';file_put_contents('log.txt', 'start-proceed:'.date('y-m-d h:i:s'), file_append);sleep(10);file_put_contents('log.txt', 'end-time:'.date('y-m-d h:i:s'), file_append);
从输出结果可看到,页面打印完program start...,输出第一行到 log.txt 后会话就返回了,所以后面的 debug... 不会在浏览器上显示,而 log.txt 文件能完整地接收到三个完成时间。
2.使用 fsockopen()
使用 fsockopen() 打开一个网络连接或者一个unix套接字连接,再用 stream_set_blocking() 非阻塞模式请求:
$fp = fsockopen("www.example.com", 80, $errno, $errstr, 30);if (!$fp) { die('error fsockopen');}// 转换到非阻塞模式stream_set_blocking($fp, 0);$http = "get /save.php / http/1.1\r\n";$http .= "host: www.example.com\r\n";$http .= "connection: close\r\n\r\n";fwrite($fp, $http);fclose($fp);
3.使用 curl
利用curl中的 curl_multi_* 函数发送异步请求
$mh = curl_multi_init();$ch = curl_init();curl_setopt($ch, curlopt_url, "http://localhost/");curl_multi_add_handle($mh, $ch);curl_multi_exec($mh, $active);curl_close($ch);curl_multi_remove_handle($mh, $ch);curl_multi_close($mh);echo "end\n";
4.使用 gearman/swoole 扩展
gearman 是一个具有 php 扩展的分布式异步处理框架,能处理大批量异步任务。
swoole 最近很火,有很多异步方法,使用简单。
5.使用缓存和队列
使用redis等缓存、队列,将数据写入缓存,使用后台计划任务实现数据异步处理。
这个方法在常见的大流量架构中应该很常见吧
6.调用系统命令
极端的情况下,可以调用系统命令,可以将数据传给后台任务执行,个人感觉不是很高效。
$cmd = 'nohup php ./processd.php $somevar >/dev/null &';`$cmd`
7.使用 pcntl_fork()
安装 pcntl 扩展,使用 pcntl_fork() 生成子进程异步执行任务,个人觉得是最方便的,但也容易出现僵尸进程。
$pid = pcntl_fork()if ($pid == 0) { child_func(); //子进程函数,主进程运行} else { father_func(); //主进程函数}echo "process " . getmypid() . " get to the end.\n"; function father_func() { echo "father pid is " . getmypid() . "\n";}function child_func() { sleep(6); echo "child process exit pid is " . getmypid() . "\n"; exit(0);}
以上就是php异步非阻塞实现方法的详细内容。
其它类似信息

推荐信息