问题是,在很长的过程中,无论当前是否已连接客户端浏览器,PHP脚本都会继续执行。如果客户端终止了对脚本的Ajax调用,那么脚本是否也可能在服务器上终止?
正如@metadings指出的那样,php确实具有一个函数来检查连接中止,该连接名为connection_aborted()。如果连接终止,它将返回1,否则返回0。
在漫长的服务器端过程中,用户可能需要了解客户端是否与服务器断开连接或他已经关闭了浏览器,然后服务器才能安全地关闭该过程。
尤其是在应用程序使用php会话的情况下,如果即使客户端断开连接后,我们仍保持长时间运行,则服务器将对此会话无响应。来自同一客户端的任何其他请求都将等待,直到之前的进程完全执行。这种情况的原因是,进程运行时会话文件被锁定。但是,您可以专心地调用session_write_close()方法将其解锁。但这并非在所有情况下都是可行的,可能是需要在过程结束时向会话写一些东西。
现在,如果我们仅在循环中调用connection_aborted(),则无论连接是否关闭,它将始终返回0。
0表示连接不会中断。这是误导。但是,经过重新研究和实验,如果发现php 中的 输出缓冲区 是原因。
首先,为了检查是否中止,循环中的开发人员必须通过回显一些文本将一些输出发送到客户端。例如:
print " ";
由于该进程仍在运行,因此不会将输出发送到客户端。现在,要发送输出,我们需要刷新输出缓冲区。
flush (); ob_flush ();
然后,如果我们检查是否中止,它将给出正确的结果。
if (connection_aborted () != 0) { die(); }
以下是工作示例,即使您正在使用PHP会话,这也将起作用:
session_start (); ignore_user_abort ( TRUE ); file_put_contents ( "con-status.txt", "Process started..\n\n" ); for($i = 1; $i <= 15; $i ++) { print " "; file_put_contents ( "con-status.txt", "Running process unit $i \n", FILE_APPEND ); sleep ( 1 ); // Send output to client flush (); ob_flush (); // Check for connection abort if (connection_aborted () != 0) { file_put_contents ( "con-status.txt", "\nUser terminated the process", FILE_APPEND ); die (); } } file_put_contents ( "con-status.txt", "\nAll units completed.", FILE_APPEND );
编辑07-APR-2017
如果有人在Windows上使用Fast-Cgi,则当使用以下代码终止连接时,他实际上可以从内存中终止CGI线程:
if (connection_aborted () != 0) { apache_child_terminate(); exit; }