PHP中register_shutdown_function函数的基础介绍与用法详解


Posted in PHP onNovember 28, 2017

前言

最近在看《PHP核心技术与最佳实践》,里面有使用到一个函数,register_shutdown_function,由于之前没有用过该函数,就去查了一下资料,就觉得是个很实用的函数,所以这里写一下这个函数的用法。下面话不多说了,来一起看看详细的介绍吧。

1. 函数说明

定义:该函数是来注册一个会在PHP中止时执行的函数

参数说明:

void register_shutdown_function ( callable $callback [, mixed $parameter [, mixed $... ]] )

注册一个 callback ,它会在脚本执行完成或者 exit() 后被调用。

callback:待注册的中止回调

parameter:可以通过传入额外的参数来将参数传给中止函数

2. PHP中止的情况

PHP中止的情况有三种:

  • 执行完成
  • exit/die导致的中止
  • 发生致命错误中止

a. 第一种情况,执行完成

<?php 
function test() 
{ 
 echo '这个是中止方法test的输出'; 
} 
 
register_shutdown_function('test'); 
 
echo 'before' . PHP_EOL;

运行:

before 
这个是中止方法test的输出

注意:输出的顺序,等执行完成了之后才会去执行register_shutdown_function的中止方法test

b. 第二种情况,exit/die导致的中止

<?php 
function test() 
{ 
 echo '这个是中止方法test的输出'; 
} 
 
register_shutdown_function('test'); 
 
echo 'before' . PHP_EOL; 
exit(); 
echo 'after' . PHP_EOL;

运行:

before 
这个是中止方法test的输出

后面的after并没有输出,即exit或者是die方法导致提前中止。

c. 第三种情况,发送致命错误中止

<?php 
function test() 
{ 
 echo '这个是中止方法test的输出'; 
} 
 
register_shutdown_function('test'); 
 
echo 'before' . PHP_EOL; 
 
// 这里会发生致命错误 
$a = new a(); 
 
echo 'after' . PHP_EOL;

运行:

before 
 
Fatal error: Uncaught Error: Class 'a' not found in D:\laragon\www\php_book\test.php on line 12 
 
Error: Class 'a' not found in D:\laragon\www\php_book\test.php on line 12 
 
Call Stack: 
 0.0020  360760 1. {main}() D:\laragon\www\php_book\test.php:0 
 
这个是中止方法test的输出

后面的after也是没有输出,致命错误导致提前中止了。

3. 参数

第一个参数支持以数组的形式来调用类中的方法,第二个以及后面的参数都是可以当做额外的参数传给中止方法。

<?php 
 
class Shutdown 
{ 
 public function stop() 
 { 
  echo "这个是stop方法的输出"; 
 } 
} 
 
// 当PHP终止的时候(执行完成或者是遇到致命错误中止的时候)会调用new Shutdown的stop方法 
register_shutdown_function([new Shutdown(), 'stop']); 
 
// 将因为致命错误而中止 
$a = new a(); 
 
// 这一句并没有执行,也没有输出 
echo '必须终止';

也可以在类中执行:

<?php 
 
class TestDemo { 
 public function __construct() 
 { 
  register_shutdown_function([$this, "f"], "hello"); 
 } 
 
 public function f($str) 
 { 
  echo "class TestDemo->f():" . $str; 
 } 
} 
 
$demo = new TestDemo(); 
echo 'before' . PHP_EOL; 
 
/** 
运行: 
before 
class TestDemo->f():hello 
 */

4. 同时调用多个

可以多次调用 register_shutdown_function,这些被注册的回调会按照他们注册时的顺序被依次调用。

不过注意的是,如果在第一个注册的中止方法里面调用exit方法或者是die方法的话,那么其他注册的中止回调也不会被调用。
代码:

<?php 
/** 
 * 可以多次调用 register_shutdown_function,这些被注册的回调会按照他们注册时的顺序被依次调用。 
 * 注意:如果你在f方法(第一个注册的方法)里面调用exit方法或者是die方法的话,那么其他注册的中止回调也不会被调用 
 */ 
 
/** 
 * @param $str 
 */ 
function f($str) { 
 echo $str . PHP_EOL; 
 
 // 如果下面调用exit方法或者是die方法的话,其他注册的中止回调不会被调用 
 // exit(); 
} 
 
// 注册第一个中止回调f方法 
register_shutdown_function("f", "hello"); 
 
class TestDemo { 
 public function __construct() 
 { 
  register_shutdown_function([$this, "f"], "hello"); 
 } 
 
 public function f($str) 
 { 
  echo "class TestDemo->f():" . $str; 
 } 
} 
 
$demo = new TestDemo(); 
echo 'before' . PHP_EOL; 
 
/** 
运行: 
before 
hello 
class TestDemo->f():hello 
 
注意:如果f方法里面调用了exit或者是die的话,那么最后的class TestDemo->f():hello不会输出 
 */

5. 用处

该函数的作用:

析构函数:在PHP4的时候,由于类不支持析构函数,所以这个函数经常用来模拟实现析构函数

致命错误的处理:使用该函数可以用来捕获致命错误并且在发生致命错误后恢复流程处理

代码如下:

<?php 
/** 
 * register_shutdown_function,注册一个会在php中止时执行的函数,中止的情况包括发生致命错误、die之后、exit之后、执行完成之后都会调用register_shutdown_function里面的函数 
 * Created by PhpStorm. 
 * User: Administrator 
 * Date: 2017/7/15 
 * Time: 17:41 
 */ 
 
class Shutdown 
{ 
 public function stop() 
 { 
  echo 'Begin.' . PHP_EOL; 
  // 如果有发生错误(所有的错误,包括致命和非致命)的话,获取最后发生的错误 
  if (error_get_last()) { 
   print_r(error_get_last()); 
  } 
 
  // ToDo:发生致命错误后恢复流程处理 
 
  // 中止后面的所有处理 
  die('Stop.'); 
 } 
} 
 
// 当PHP终止的时候(执行完成或者是遇到致命错误中止的时候)会调用new Shutdown的stop方法 
register_shutdown_function([new Shutdown(), 'stop']); 
 
// 将因为致命错误而中止 
$a = new a(); 
 
// 这一句并没有执行,也没有输出 
echo '必须终止';

运行:

Fatal error: Uncaught Error: Class 'a' not found in D:\laragon\www\php_book\1_23_register_shutdown.php on line 31 
 
Error: Class 'a' not found in D:\laragon\www\php_book\1_23_register_shutdown.php on line 31 
 
Call Stack: 
 0.0060  362712 1. {main}() D:\laragon\www\php_book\1_23_register_shutdown.php:0 
 
Begin. 
Array 
( 
 [type] => 1 
 [message] => Uncaught Error: Class 'a' not found in D:\laragon\www\php_book\1_23_register_shutdown.php:31 
Stack trace: 
#0 {main} 
 thrown 
 [file] => D:\laragon\www\php_book\1_23_register_shutdown.php 
 [line] => 31 
) 
Stop.

注意:PHP7中新增了Throwable异常类,这个类可以捕获致命错误,即可以使用try...catch(Throwable $e)来捕获致命错误,代码如下:

<?php 
 
try { 
 // 将因为致命错误而中止 
 $a = new a(); 
 
 // 这一句并没有执行,也没有输出 
 echo 'end'; 
} catch (Throwable $e) { 
 print_r($e); 
 echo $e->getMessage(); 
}

运行:

Error Object 
( 
 [message:protected] => Class 'a' not found 
 [string:Error:private] => 
 [code:protected] => 0 
 [file:protected] => C:\laragon\www\php_book\throwable.php 
 [line:protected] => 5 
 [trace:Error:private] => Array 
  ( 
  ) 
 
 [previous:Error:private] => 
 [xdebug_message] => 
Error: Class 'a' not found in C:\laragon\www\php_book\throwable.php on line 5 
 
Call Stack: 
 0.0000  349856 1. {main}() C:\laragon\www\php_book\throwable.php:0 
 
) 
Class 'a' not found

这样的话,PHP7中使用Throwable来捕获的话比使用register_shutdown_function这个函数来得更方便,也更推荐Throwable。

注意:Error类也是可以捕获到致命错误,不过Error只能捕获致命错误,不能捕获异常Exception,而Throwable是可以捕获到错误和异常的,所以更推荐。

6.巧用register_shutdown_function判断php程序是否执行完

还有一种应用场景就是:要做一个消费队列,因为某条有问题的数据导致致命错误,如果这条数据不处理掉,那么整个队列都会导致瘫痪的状态,这样可以用以下方法来解决。即:如果捕获到有问题的数据导致错误,则在回调函数中将这条数据处理掉就可以了。

php范例参考与解析:

<?php

register_shutdown_function('myFun'); //放到最上面,不然如果下面有致命错误,就不会调用myFun了。
$execDone = false; //程序是否成功执行完(默认为false)

/**
********************* 业务逻辑区*************************
*/
$tas = 3;
if($tas == 3)
{
new daixiaorui();
}

/**
********************* 业务逻辑结束*************************
*/
$execDone = true; //由于程序由上至下执行,因此当执行到此后,则证明逻辑没有出现致命的错误。

function myFun()
{
global $execDone;
if($execDone === false)
{
file_put_contents("E:/myMsg.txt", date("Y-m-d H:i:s")."---error: 程序执行出错。\r\n", FILE_APPEND);
/******** 以下可以做一些处理 ********/
}
}

总结

register_shutdown_function这个函数主要是用在处理致命错误的后续处理上(PHP7更推荐使用Throwable来处理致命错误),不过缺点也很明显,只能处理致命错误Fatal error,其他的错误包括最高错误Parse error也是没办法处理的。

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

PHP 相关文章推荐
我的论坛源代码(四)
Oct 09 PHP
一个PHP并发访问实例代码
Sep 06 PHP
php生成局部唯一识别码LUID的代码
Oct 06 PHP
浅析PHP的静态成员函数效率更高的原因
Jun 13 PHP
codeigniter中view通过循环显示数组数据的方法
Mar 20 PHP
CI配置多数据库访问的方法
Mar 28 PHP
PHP中单例模式与工厂模式详解
Feb 17 PHP
PHP 文件锁与进程锁的使用示例
Aug 07 PHP
PHP排序算法之快速排序(Quick Sort)及其优化算法详解
Apr 21 PHP
php接口实现拖拽排序功能
Apr 23 PHP
PHP面向对象程序设计之对象的遍历操作示例
Jun 12 PHP
Laravel jwt 多表(多用户端)验证隔离的实现
Dec 18 PHP
PHP命令空间namespace及use的用法小结
Nov 27 #PHP
Laravel 批量更新多条数据的示例
Nov 27 #PHP
PHP开发实现微信退款功能示例
Nov 25 #PHP
PHP微信企业号开发之回调模式开启与用法示例
Nov 25 #PHP
PHP递归实现汉诺塔问题的方法示例
Nov 25 #PHP
PHP基于curl post实现发送url及相关中文乱码问题解决方法
Nov 25 #PHP
php图片合成方法(多张图片合成一张)
Nov 25 #PHP
You might like
discuz的php防止sql注入函数
2011/01/17 PHP
php ajax 静态分页过程形式
2011/09/02 PHP
用PHP+MySQL搭建聊天室功能实例代码
2012/08/20 PHP
ecshop添加菜单及权限分配问题
2017/11/21 PHP
gearman管理工具GearmanManager的安装与php使用方法示例
2020/02/27 PHP
Stop SQL Server
2007/06/21 Javascript
js 控制下拉菜单刷新的方法
2013/03/03 Javascript
把input初始值不写value的具体实现方法
2013/07/04 Javascript
QQ登录背景闪动效果附效果演示源码下载
2015/09/22 Javascript
javascript同步服务器时间和同步倒计时小技巧
2015/09/24 Javascript
JavaScript的函数式编程基础指南
2016/03/19 Javascript
基于JS实现移动端访问PC端页面时跳转到对应的移动端网页
2020/12/24 Javascript
jQuery模拟Marquee实现无缝滚动效果完整实例
2016/09/29 Javascript
在js代码拼接dom对象到页面上去的模板总结(必看)
2017/02/14 Javascript
vue2 前后端分离项目ajax跨域session问题解决方法
2017/04/27 Javascript
nodejs mysql 实现分页的方法
2017/06/06 NodeJs
详解JavaScript按概率随机生成事件
2017/08/02 Javascript
谈谈JS中的!!
2017/12/07 Javascript
vue实现循环切换动画
2018/10/17 Javascript
jQuery Migrate 插件用法实例详解
2019/05/22 jQuery
JavaScript实现沿五角星形线摆动的小圆实例详解
2020/07/28 Javascript
python基于windows平台锁定键盘输入的方法
2015/03/05 Python
用python处理图片实现图像中的像素访问
2018/05/04 Python
django celery redis使用具体实践
2019/04/08 Python
python实现接口并发测试脚本
2019/06/25 Python
Pandas之Fillna填充缺失数据的方法
2019/06/25 Python
python异常处理和日志处理方式
2019/12/24 Python
Python Spyder 调出缩进对齐线的操作
2021/02/26 Python
美国销售第一的智能手机和平板电脑保护壳:OtterBox
2017/12/21 全球购物
世界上最好的旅行夹克:BauBax
2018/12/23 全球购物
大专计算机个人求职的自我评价
2013/10/21 职场文书
英语专业毕业生自我鉴定
2013/11/09 职场文书
表彰会主持词
2014/03/26 职场文书
北京申奥口号
2014/06/19 职场文书
社保转移委托书范本
2014/10/08 职场文书
「天才王子的赤字国家重生术」妮妮姆·拉雷粘土人开订
2022/03/21 日漫