PHP 返回结果给前端/ajax后,在后台继续执行代码的方法
. 问题背景
在实际项目开发中,遇到一个问题:
前端通过 Ajax 请求后台 PHP API 接口,执行多文件的打包下载操作,该请求由于需要更新大量的数据(日志、统计等信息)到数据库且还需要执行较大的磁盘IO操作,导致该请求很耗时间。由于前端页面的更新需要快速响应,因此需要 PHP 快速返回计算结果,然后后台继续执行余下的操作。
2. 解决方法
exit() 之后还能继续执行代码的方法有 析构函数 __destruct() 以及 register_shutdown_function() (记日志或者xhprof等性能分析等有一定耗时的代码),但针对 ajax 请求并不能立即返回
考虑到 HTTP 请求协议中可以通过 flush() 进行局部内容输出,立即返回请求结果给前端,再将耗时操作继续执行,即通过该技术解决问题
{ $rs = ['code' => 0, 'msg' => 'ok', 'data' => true]; ob_end_clean(); ob_start(); //----------------------------------------------------------------------------------- //Windows服务器需要加上这行。 //echo str_repeat(" ",4096); echo json_encode($res);//返回结果给ajax //----------------------------------------------------------------------------------- // get the size of the output $size = ob_get_length(); // send headers to tell the browser to close the connection header("Content-Length: $size"); header('Connection: close'); header("HTTP/1.1 200 OK"); header("Content-Type: application/json;charset=utf-8"); ob_end_flush(); if(ob_get_length()) ob_flush(); flush(); if (function_exists("fastcgi_finish_request")) { // yii或yaf默认不会立即输出,加上此句即可(前提是用的fpm) fastcgi_finish_request(); // 响应完成, 立即返回到前端,关闭连接 } /******** background process starts here ********/ ignore_user_abort(true);//在关闭连接后,继续运行php脚本 /******** background process ********/ set_time_limit(0); //no time limit,不设置超时时间(根据实际情况使用) /******** Rest of your code starts here ********/ //继续运行的代码 ... ...
对于长时间运行的代码可以考虑使用消息队列方式替代 HTTP 的 flush 特性 — 推荐
后台异步调用 HTTP 请求的方法可通过 fsockopen 实现
As of August 2012, all browsers seem to show an all-or-nothing approach to buffering. In other words, while php is operating, no content can be shown.
In particular this means that the following workarounds listed further down here are ineffective:
ob_flush (), flush () in any combination with other output buffering functions;
changes to php.ini involving setting output_buffer and/or zlib.output_compression to 0 or Off;
setting Apache variables such as "no-gzip" either through apache_setenv ()or through entries in .htaccess.
So, until browsers begin to show buffered content again, the tips listed here are moot.
————————————————
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
原文链接:https://blog.csdn.net/aoshilang2249/article/details/82772329