PHP set_error_handler()函数使用详解(示例)


Posted in PHP onNovember 12, 2013

我们写程序,难免会有问题(是经常会遇到问题 ),而PHP遇到错误时,就会给出出错脚本的位置、行数和原因。有很多人说,这并没有什么大不了。确实,在调试程序阶段,这确实是没啥的,而且我认为给出错误路径是必要的。
但泄露了实际路径的后果是不堪设想的,对于某些入侵者,这个信息可是非常重要,而事实上现在有很多的服务器都存在这个问题。有些网管干脆把PHP配置文件中的display_errors设置为Off来解决(貌似我们就是这样做的),但本人认为这个方法过于消极。
有些时候,我们的确需要PHP返回错误的信息以便调试。而且在出错时也可能需要给用户一个交待,甚至导航到另一页面。那么,有啥解决办法呢?
set_error_handler()
PHP从4.1.0开始提供了自定义错误处理句柄的功能函数set_error_handler(),但很少数脚本编写者知道。set_error_handler这个函数可以很好地防止错误路径泄露,当然还有其它更多的作用。
可以用来屏蔽错误。 出现错误一来会把一些信息暴漏给用户,极有可能成为黑客攻击你网站的工具。 二来让用户觉得你的水平很挫。
可以记下错误的信息, 及时发现一些生产环境的出现的问题。
可以做相应的处理, 出错的时候可以显示跳转到预先定义好的出错页面,提供更好的用户体验。
可以作为调试工具, 一些时候必须在生产环境调试一些东西, 但又不想影响正在使用的用户。
set_error_handler的使用方法如下:

string set_error_handler ( callback error_handler [, int error_types])

现在我们就用自定义的错误处理把实际路径过滤掉。假设有一个变量$admin,我们是用来判断访问者是否是管理员的(可以通过IP或者登录的用户id来做这个判断)
//admin为管理员的身份判定,true为管理员。  
//自定义的错误处理函数一定要有这4个输入变量$errno,$errstr,$errfile,$errline,否则无效。  
function my_error_handler($errno,$errstr,$errfile,$errline)  
{  
    //如果不是管理员就过滤实际路径  
    if(!admin)  
    {  
        $errfile=str_replace(getcwd(),"",$errfile);  
        $errstr=str_replace(getcwd(),"",$errstr);  
    }  
    switch($errno)  
    {  
        case E_ERROR:  
        echo "ERROR: [ID $errno] $errstr (Line: $errline of $errfile) \n";  
        echo "程序已经停止运行,请联系管理员。";  
        //遇到Error级错误时退出脚本  
        exit;  
        break;          case E_WARNING:  
        echo "WARNING: [ID $errno] $errstr (Line: $errline of $errfile) \n";  
        break;  
        default:  
        //不显示Notice级的错误  
        break;  
    }  
} 

这样就自定义了一个错误处理函数,那么怎么把错误的处理交给这个自定义函数呢?
// 应用到类  
set_error_handler(array(&$this,"appError"));  //示例的做法  
set_error_handler("my_error_handler"); 

so easy,这样,就可以很好地解决安全和调试方便的矛盾了。而且你还可以花点心思,使错误提示更加美观以配合网站的风格。
原作者给出了两点需要注意的地方,我也放出来吧,希望引起广大同胞们的注意:
1、E_ERROR、E_PARSE、E_CORE_ERROR、E_CORE_WARNING、 E_COMPILE_ERROR、E_COMPILE_WARNING是不会被这个句柄处理的,也就是会用最原始的方式显示出来。不过出现这些错误都是编 译或PHP内核出错,在通常情况下不会发生。
2、使用set_error_handler()后,error_reporting ()将会失效。也就是所有的错误(除上述的错误)都会交给自定义的函数处理。
最后,出一个示例
//先定义一个函数,也可以定义在其他的文件中,再用require()调用  
function myErrorHandler($errno, $errstr, $errfile, $errline)  
{  

 //为了安全起见,不暴露出真实物理路径,下面两行过滤实际路径  
    $errfile=str_replace(getcwd(),"",$errfile);  
    $errstr=str_replace(getcwd(),"",$errstr);  
    switch ($errno) {  
    case E_USER_ERROR:  
     echo "<b>My ERROR</b> [$errno] $errstr<br />\n";  
        echo "  Fatal error on line $errline in file $errfile";  
        echo ", PHP " . PHP_VERSION . " (" . PHP_OS . ")<br />\n";  
        echo "Aborting...<br />\n";  
        exit(1);  
        break;  
    case E_USER_WARNING:  
        echo "<b>My WARNING</b> [$errno] $errstr<br />\n";  
        break;  
    case E_USER_NOTICE:  
        echo "<b>My NOTICE</b> [$errno] $errstr<br />\n";  
        break;  
    default:  
        echo "Unknown error type: [$errno] $errstr<br />\n";  
        break;  
    }  
    /* Don't execute PHP internal error handler */  
    return true;  
}  
//下面开始连接MYSQL服务器,我们故意指定MYSQL端口为3333,实际为3306。  
$link_id=@mysql_pconnect("localhost:3333","root","password");  
set_error_handler(myErrorHandler);  
if (!$link_id) {  
    trigger_error("出错了", E_USER_ERROR);  
} 
PHP 相关文章推荐
用PHP实现Ftp用户的在线管理的代码
Mar 06 PHP
随时给自己贴的图片加文字的php代码
Mar 08 PHP
在php MYSQL中插入当前时间
Apr 06 PHP
30个php操作redis常用方法代码例子
Jul 05 PHP
浅析THINKPHP的addAll支持的最大数据量
Feb 03 PHP
PHP批量查询WordPress留言者E-mail地址实现方法
Feb 15 PHP
php校验表单检测字段是否为空的方法
Mar 20 PHP
PHP Ajax实现无刷新附件上传
Aug 17 PHP
php arsort 数组降序排序详细介绍
Nov 17 PHP
php判断str字符串是否是xml格式数据的方法示例
Jul 26 PHP
基于win2003虚拟机中apache服务器的访问
Aug 01 PHP
PHP实现的策略模式简单示例
Aug 25 PHP
php inc文件使用的风险和注意事项
Nov 12 #PHP
php防止SQL注入详解及防范
Nov 12 #PHP
php session劫持和防范的方法
Nov 12 #PHP
php后门URL的防范
Nov 12 #PHP
php打开远程文件的方法和风险及解决方法
Nov 12 #PHP
php使用exec shell命令注入的方法讲解
Nov 12 #PHP
PHP使用PHPMailer发送邮件的简单使用方法
Nov 12 #PHP
You might like
php 上一篇,下一篇文章实现代码与原理说明
2010/05/09 PHP
PHP实现Google plus的好友拖拽分组效果
2016/10/21 PHP
PHP 文件上传限制问题
2019/09/01 PHP
PHP生成随机字符串实例代码(字母+数字)
2019/09/11 PHP
PHP程序员简单的开展服务治理架构操作详解(一)
2020/05/14 PHP
IE浏览器兼容Firefox的JS脚本的代码
2008/10/23 Javascript
让网页根据不同IE版本显示不同的内容
2009/02/08 Javascript
JS与框架页的操作代码
2010/01/17 Javascript
juqery 学习之六 CSS--css、位置、宽高
2011/02/11 Javascript
javascript中的=等号个数问题两个跟三个有什么区别
2013/10/23 Javascript
jquery showModelDialog的使用方法示例详解
2013/11/19 Javascript
jQuery select表单提交省市区城市三级联动核心代码
2014/06/09 Javascript
jQuery中height()方法用法实例
2014/12/24 Javascript
深入分析下javascript中的[]()+!
2015/07/07 Javascript
jQuery post数据至ashx实例详解
2016/11/18 Javascript
BootStrap Fileinput初始化时的一些参数
2016/12/30 Javascript
使用vue和datatables进行表格的服务器端分页实例代码
2017/06/07 Javascript
vue的基本用法与常见指令
2017/08/15 Javascript
JS实现的计数排序与基数排序算法示例
2017/12/04 Javascript
Vuex的实战使用详解
2019/10/31 Javascript
介绍Python中的文档测试模块
2015/04/28 Python
Python栈类实例分析
2015/06/15 Python
PyChar学习教程之自定义文件与代码模板详解
2017/07/17 Python
wxPython的安装图文教程(Windows)
2017/12/28 Python
Python元字符的用法实例解析
2018/01/17 Python
python可视化实现代码
2019/01/15 Python
Python实现钉钉发送报警消息的方法
2019/02/20 Python
Python3实现定时任务的四种方式
2019/06/03 Python
python的pstuil模块使用方法总结
2019/07/26 Python
详解在Python中以绝对路径或者相对路径导入文件的方法
2019/08/30 Python
服装设计专业毕业生求职信
2014/04/09 职场文书
公司股东合作协议书
2014/09/14 职场文书
交通肇事罪辩护词
2015/05/21 职场文书
幼师必备:幼儿园期末教师评语50条
2019/11/01 职场文书
你真的了解redis为什么要提供pipeline功能
2021/06/22 Redis
python基础之类属性和实例属性
2021/10/24 Python