试想如下情况。如果你的用户正在执行一个需要非常长的执行时间的操作。他点了执行了之后,浏览器就开始蛋疼地转。如果执行5分钟,你猜他会干啥,显然会觉得什么狗屎垃圾站,这么久都不响应,然后就给关了。当然这个是一个极端的情况。但是我们不乏这种需求,需要用户关闭请求之后也能够做事情。脚本继续执行。再考虑另外一种情况。如果我们现在正在做一个非常精确的事情,用户一旦断开(异常或者是正常)之后必须要执行一些操作,比如他掉线了,我们需要把他的在线状态清空。
基于有这样的需求,我找到了 ignore_user_abort 这个函数。
当调用 ignore_user_abort(1) 时, 就设定了忽略用户退出这个标志位,也就是不管怎样,也要把程序执行到底,除非在程序中调用了 exit。(PHPINI也可以设置这个)
还有另一个函数函register_shutdown_function,它可以注册一个函数或方法,在程序退出的时候调用,有些类似于 javascript 里面的 onunload 和 onbeforeunload 事件。
ignore_user_abort(1)很简单,设置之后一直要直到你的脚本执行完成,当然这个也是一个非常危险的操作,可能会产生众多的僵尸进程,处理不当的话。
手册上如下描述:
注意: 以下内容对 PHP 3.0.7 及更高版本适用。
在 PHP 内部,系统维护着连接状态,其状态有三种可能的情况:
0 - NORMAL(正常)
1 - ABORTED(异常退出)
2 - TIMEOUT(超时)
当 PHP 脚本正常地运行 NORMAL 状态时,连接为有效。当远程客户端中断连接时,ABORTED 状态的标记将会被打开。远程客户端连接的中断通常是由用户点击 STOP 按钮导致的。当连接时间超过 PHP 的时限(请参阅 set_time_limit() 函数)时,TIMEOUT 状态的标记将被打开。
可以决定脚本是否需要在客户端中断连接时退出。有时候让脚本完整地运行会带来很多方便,即使没有远程浏览器接受脚本的输出。默认的情况是当远程客户 端连接中断时脚本将会退出。该处理过程可由 php.ini 的 ignore_user_abort 或由 Apache .conf 设置中对应的“php_value ignore_user_abort”以及 ignore_user_abort() 函数来控制。如果没有告诉 PHP 忽略用户的中断,脚本将会被中断,除非通过 register_shutdown_function() 设置了关闭触发函数。通过该关闭触发函数,当远程用户点击 STOP 按钮后,脚本再次尝试输出数据时,PHP 将会检测到连接已被中断,并调用关闭触发函数。
脚本也有可能被内置的脚本计时器中断。默认的超时限制为 30 秒。这个值可以通过设置 php.ini 的 max_execution_time 或 Apache .conf 设置中对应的“php_value max_execution_time”参数或者 set_time_limit() 函数来更改。当计数器超时的时候,脚本将会类似于以上连接中断的情况退出,先前被注册过的关闭触发函数也将在这时被执行。在该关闭触发函数中,可以通过调 用 connection_status() 函数来检查超时是否导致关闭触发函数被调用。如果超时导致了关闭触发函数的调用,该函数将返回 2。
需要注意的一点是 ABORTED 和 TIMEOUT 状态可以同时有效。这在告诉 PHP 忽略用户的退出操作时是可能的。PHP 将仍然注意用户已经中断了连接但脚本仍然在运行的情况。如果到了运行的时间限制,脚本将被退出,设置过的关闭触发函数也将被执行。在这时会发现函数 connection_status() 返回 3。
OK,看这个函数非常简单也很容易理解。注意上面一句话: register_shutdown_function() 设置了关闭触发函数。通过该关闭触发函数,当远程用户点击 STOP 按钮后,脚本再次尝试输出数据时,PHP 将会检测到连接已被中断,并调用关闭触发函数。
再去查一查 :register_shutdown_function() 如下描述:
Registers a callback
to be executed after script execution finishes or is called.
也就是说脚本执行完成的时候或者使用exit()的时候调用。
但是异常中断算什么。
再看加粗的那句话,当脚本尝试输出数据的时候PHP会检测到用户链接已经被中断,明白了,只要我输出就知道他是否中断,前几天看得flush函数又派上用场了,我只要一直flush就可以直到他中断没有。然后调用退出的回调函数。这个也是之前写的聊天状态管理为什么死循环里面写echo 1;flush();的原因。