PHP中的错误处理、异常处理机制详解
在编写PHP程序时,错误处理是一个重要的部分。如果程序中缺少错误检测代码,那么看上去很不专业,也为安全风险敞开了大门
例:
<?php
$a = fopen('test.txt','r');
//这里并没有对文件进行判断就打开了,如果文件不存在就会报错
?>
那么正确的写法应该如下:
<?php
if (file_exists('test.txt')) {
$f = fopen('test.txt', 'r');
// 使用完后关闭
fclose($f);
}
?>
一、PHP错误处理的三种方式
A、简单的die()语句;
等价于exit();
例:
if (!file_exists('aa.txt')) {
die('文件不存在');
} else {
// 执行操作
}
// 如果上面die()被触发,那么这里echo接不被执行
echo 'ok';
简洁写法:
file_exits('aaa.txt') or die('文件不存在');
echo 'ok';
B、自定义错误和错误触发器
1、错误处理器(自定义错误,一般用于语法错误处理)
创建自定义错误函数(处理器),该函数必须有能力处理至少两个参数(error_level和errormessage),但是可以接受最多五个参数(error_file、error_line、error_context)
语法:
function error_function($error_level, $error_message, $error_file, $error_line, $error_context)
// 创建好后还需要改写set_error_handler();函数
set_error_handler('error_function', E_WARNING); // 这里error_function对应上面创建的自定义处理器名,第二个参数为使用自定义错误处理器的错误级别;
错误报告级别(了解即可)
这些错误报告级别是错误处理程序旨在处理的错误的不同的类型:
值 | 常量 | 描述 |
---|---|---|
2 | E_WARNING | 非致命的 run-time 错误。不暂停脚本执行。 |
8 | E_NOTICE | Run-time 通知。脚本发现可能有错误发生,但也可能在脚本正常运行时发生。 |
256 | E_USER_ERROR | 致命的用户生成的错误。这类似于程序员使用 PHP 函数 trigger_error() 设置的 E_ERROR。 |
512 | E_USER_WARNING | 非致命的用户生成的警告。这类似于程序员使用 PHP 函数 trigger_error() 设置的 E_WARNING。 |
1024 | E_USER_NOTICE | 用户生成的通知。这类似于程序员使用 PHP 函数 trigger_error() 设置的 E_NOTICE。 |
4096 | E_RECOVERABLE_ERROR | 可捕获的致命错误。类似 E_ERROR,但可被用户定义的处理程序捕获。(参见 set_error_handler()) |
8191 | E_ALL | 所有错误和警告,除级别 E_STRICT 以外。(在 PHP 6.0,E_STRICT 是 E_ALL 的一部分) |
2、错误触发器(一般用于处理逻辑上的错误)
需求:比如要接收一个年龄,如果数字大于120,就认为是一个错误
传统方法:
<?php
if ($age > 120) {
echo '年龄错误';
exit();
}
?>
使用触发器:
<?php
if ($age > 120) {
// trigger_error('错误信息'[,'错误等级']); 这里错误等级为可选项,用于定义该错误的级别
// 用户定义的级别包含以下三种:E_USER_WARNING 、E_USER_ERROR 、E_USER_NOTICE
trigger_error('年龄错误'); // 这里是调用的系统默认的错误处理方式,我们也可以用自定义处理器
} /**
* 自定义处理器,与上面相同
*/
function myerror($error_level, $error_message) {
echo 'error text';
} // 同时需要改变系统默认的处理函数
set_error_handler('myerror', E_USER_WARNING); // 同上面,第一个参数为自定义函数的名称,第二个为错误级别【这里的错误级别通常为以下三种:E_USER_WARNING 、E_USER_ERROR 、E_USER_NOTICE】
// 现在再使用trigger_error就可以使用自定义的错误处理函数了
?>
练习题:
<?php
date_default_timezone_set('PRC');
function myerror($error_level, $error_message) {
$info = "错误号:$error_level";
$info .= "错误信息:$error_message";
$info .= '发生时间:' . date('Y-m-d H:i:s');
$filename = 'aa.txt';
if (!$fp = fopen($filename, 'a')) {
echo '创建文件' . $filename . '失败';
}
if (is_writeable($filename)) {
if (!fwrite($fp, $info)) {
echo '写入文件失败';
} else {
echo '已成功记录错误信息';
}
fclose($fp);
} else {
echo '文件' . $filename . '不可写';
}
exit();
} set_error_handler('myerror', E_WARNING);
$fp = fopen('aaa.txt', 'r');
?>
C、错误日志
默认的根据php.ini中error_log配置,PHP向服务器的错误记录系统或文件发送错误记录。通过使用error_log()函数可以向文件或远程目的地发送错误记录;
语法:
error_log(error[, type, destination, headers])
type部分一般用3,表示在文件后面追加错误信息,而不会覆盖原内容destination表示目的地,即存放的文件或远程目的地
如:error_log("$error_info",3,"errors.txt");
二、PHP异常处理【重点】
1、基本语法
<?php
try {
// 可能出现错误或异常的代码
//catch 捕获 Exception是PHP已定义好的异常类
} catch (Exception $e) {
// 对异常处理,方法:
//1、自己处理
//2、不处理,可以再次抛出 throw new Exception('xxx');
}
?>
2、处理处理程序应当包括:
- try - 使用异常的函数应该位于 "try" 代码块内。如果没有触发异常,则代码将照常继续执行。但是如果异常被触发,会抛出一个异常;
- throw - 这里规定如何触发异常。每一个 "throw" 必须对应至少一个 "catch";
- catch - "catch" 代码块会捕获异常,并创建一个包含异常信息的对象;
让我们触发一个异常:
<?php
/**
* 创建可抛出一个异常的函数
*/
function checkNum($number) {
if ($number > 1) {
throw new Exception("Value must be 1 or below");
} return true;
} // 在 "try" 代码块中触发异常
try {
checkNum(2);
// 如果异常被抛出,那么下面一行代码将不会被输出
echo 'If you see this, the number is 1 or below';
} catch (Exception $e) {
// 捕获异常
echo 'Message: ' . $e->getMessage();
}
?>
上面代码将获得类似这样一个错误:
Message: Value must be 1 or below
例子解释:
上面的代码抛出了一个异常,并捕获了它:
- 创建 checkNum() 函数,它检测数字是否大于 1,如果是,则抛出一个异常。
- 在 "try" 代码块中调用 checkNum() 函数。
- checkNum() 函数中的异常被抛出。
- "catch" 代码块接收到该异常,并创建一个包含异常信息的对象 ($e)。
- 通过从这个 exception 对象调用 $e->getMessage(),输出来自该异常的错误消息。
不过,为了遵循“每个 throw 必须对应一个 catch”的原则,可以设置一个顶层的异常处理器来处理漏掉的错误。
set_exception_handler()函数可设置处理所有未捕获异常的用户定义函数。
<?php
/**
* 设置一个顶级异常处理器
*/
function myexception($e) {
echo 'this is top exception';
} // 修改默认的异常处理器
set_exception_handler("myexception");
try {
$i = 5;
if ($i < 10) {
throw new Exception('$i must greater than 10');
}
} catch (Exception $e) {
// 处理异常
echo $e->getMessage() . '<br/>'; // 不处理异常,继续抛出
throw new Exception('errorinfo'); // 也可以用throw $e 保留原错误信息;
}
?>
创建一个自定义的异常类
<?php
class customException extends Exception {
public function errorMessage() {
$errorMsg = 'Error on line ' . $this->getLine() . ' in ' . $this->getFile() . ': <b>' . $this->getMessage() . '</b> is not a valid E-Mail address';
return $errorMsg;
}
} // 使用
try {
throw new customException('error message');
} catch (customException $e) {
echo $e->errorMessage();
}
?>
可以使用多个catch来返回不同情况下的错误信息
<?php
try {
$i = 5;
if ($i > 0) {
throw new customException('error message'); // 使用自定义异常类处理
}
if ($i < -10) {
throw new Exception('error2'); // 使用系统默认异常处理
}
} catch (customException $e) {
echo $e->getMessage();
} catch (Exception $e1) {
echo $e1->getMessage();
}
?>
异常的规则
- 需要进行异常处理的代码应该放入 try 代码块内,以便捕获潜在的异常。
- 每个try或throw代码块必须至少拥有一个对应的 catch 代码块。
- 使用多个 catch 代码块可以捕获不同种类的异常。
- 可以在try代码内的catch 代码块中再次抛出(re-thrown)异常。
简而言之:如果抛出了异常,就必须捕获它。
延伸阅读:
PHP错误处理函数set_error_handler()的用法
PHP异常处理函数set_exception_handler()的用法
PHP中的错误处理、异常处理机制详解的更多相关文章
- java异常处理机制详解
java异常处理机制详解 程序很难做到完美,不免有各种各样的异常.比如程序本身有bug,比如程序打印时打印机没有纸了,比如内存不足.为了解决这些异常,我们需要知道异常发生的原因.对于一些常见的异常,我 ...
- SpringMVC异常处理机制详解[附带源码分析]
目录 前言 重要接口和类介绍 HandlerExceptionResolver接口 AbstractHandlerExceptionResolver抽象类 AbstractHandlerMethodE ...
- Linux内核异常处理体系结构详解(一)【转】
转自:http://www.techbulo.com/1841.html 2015年11月30日 ⁄ 基础知识 ⁄ 共 6653字 ⁄ 字号 小 中 大 ⁄ Linux内核异常处理体系结构详解(一)已 ...
- Yii中的错误及异常处理
Yii中的错误及异常处理 Yii已经默认已经在CApplication上实现了异常和错误的接管,这是通过php的set_exception_handler, set_error_handler实现的. ...
- ThinkPHP5.0源码学习之注册错误和异常处理机制
在base.php文件中,用一句代码\think\Error::register();实现错误和异常处理机制的注册. // 注册错误和异常处理机制 \think\Error::register(); ...
- PHP 注册错误和异常处理机制
注册错误和异常处理机制有三个PHP函数需要学习 1. register_shutdown_function('Bootstrap\Library\Frame::fatalError'); 2. set ...
- Linux 内存机制详解宝典
Linux 内存机制详解宝典 在linux的内存分配机制中,优先使用物理内存,当物理内存还有空闲时(还够用),不会释放其占用内存,就算占用内存的程序已经被关闭了,该程序所占用的内存用来做缓存使用,对于 ...
- iOS的消息转发机制详解
iOS开发过程中,有一类的错误会经常遇到,就是找不到所调用的方法,当然这类问题比较好解决,给当前对象或其父类对象添加该方法即可,使得编译器在编译时能正确找到该方法:或者,还有另外的方法,由于Objec ...
- epoll机制详解
epoll机制详解 大牛的详解 epoll详解 什么是epoll? epoll是为处理大批量句柄而作了改进的poll, 是性能最好的多路I/O就绪通知方法; 只有三个系统调用: epoll_creat ...
随机推荐
- Qt 之 饼图
Qt 库由许多模块组成,其中的 Qt Charts,包含了一系列图表组件. 1 饼图 (pie chart) 1.1 Charts 模块 .pro 中添加如下语句: QT += charts 头文 ...
- Hibernate第一个例子
我们先搭建这样的一个架构 里面包括实体类,实现类, 大配置, 小配置(映射文件), 以及架包 实体类我们就不重点介绍了 我们先把我们所需要用到的架包导入进来 我们先在src根目录下新建一个文件夹名为l ...
- Winform(DataGridView)控件及通过此控件中实现增删改查
DataGridView:显示数据表,通过此控件中可以实现连接数据库,实现数据的增删改查 一.后台数据绑定: List<xxx> list = new List<xxx> ...
- IntelliJ IDEA 乱码解决方案 (项目代码、控制台等)
IntelliJ IDEA 乱码解决方案 (项目代码.控制台等) 最近IDE从eclipse改成IntelliJ IDEA 了,原因是公司大部分人都在用这个IDE,而且一直推荐用,所以尝尝鲜.换的第一 ...
- python有超时的windows系统时间设置代码
手边的笔记本用久了,cmos电池可能又没电了.每次开机时间都不对,导致访问一些有https的网页会出现警告信息. 于是找了找通过python脚本设置系统时间的方法,发现了两种,其一是调用socket直 ...
- ajax实现下拉菜单无刷新加载更多
$(function() { var page = 1; var discount = $('#discount'); var innerHeight = window.innerHeight; va ...
- JS处理四舍五入函数 toFixed(n)(可取小数点后n位)
在JS中四舍五入的函数 toFixed(n) , n为要保留的小数位数. n为0~20,当n超过20的时候,JS会出错. 如果小数点前和要截取的前一位都是0时,不会按常理截取. var h ...
- iOS 关于字体根据不同屏幕尺寸等比适配的问题(zz)
http://www.jianshu.com/p/5815e81abb52 背景 去年的六月份开始了一个新的项目,此项目支持的设备从4S开始一直到6+,也就是说屏幕的尺寸最小的320x480 最大的1 ...
- Servlet和JSP
Servlet 一.Servlet 的生命周期. servlet 有良好的生存期的定义,包括加载和实例化.初始化.处理请求以及服务结束.这个生存期由javax.servlet.Servlet 接口 的 ...
- 【强烈推荐】数据库迁移利器:Migrator.Net
简介 很郁闷,写了一天的遇到LiveWriter错误,可恶啊 几年前在做项目中第一次接触到了Migrator.Net,就深深被吸引住了,至此以后在新的大项目中,我都会使用Migrator.Net来创建 ...