PHP异步非阻塞实现方法

来源:藏色散人 发布时间:2019-01-10 10:44:51 阅读量:1017

为让 PHP 在后端处理长时间任务时不阻塞,快速响应页面请求,可以有如下措施:

1.使用 fastcgi_finish_request()

如果 PHP 与 Web 服务器使用了 PHP-FPM(FastCGI 进程管理器),那通过 fastcgi_finish_request() 函数能马上结束会话,而 PHP 线程可以继续在后台运行。

1

2

3

4

5

6

7

8

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() 非阻塞模式请求:

1

2

3

4

5

6

7

8

9

10

11

$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_* 函数发送异步请求

1

2

3

4

5

6

7

8

9

$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.调用系统命令

极端的情况下,可以调用系统命令,可以将数据传给后台任务执行,个人感觉不是很高效。

1

2

$cmd = 'nohup php ./processd.php $someVar >/dev/null  &';

`$cmd`

7.使用 pcntl_fork()

安装 pcntl 扩展,使用 pcntl_fork() 生成子进程异步执行任务,个人觉得是最方便的,但也容易出现僵尸进程。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

$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 环境搭建
分享:
评论:
你还没有登录,请先