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 ...
随机推荐
- Leetcode: word search
July 6, 2015 Problem statement: Word Search Given a 2D board and a word, find if the word exists in ...
- 一个页面从输入URL到页面加载显示完成,这个过程都发生了什么?
对于网址栏的URL不同的操作方式有不同的加载资源.获取数据的方式,下面的详细过程针对"在地址栏输入URL,按enter(回车)键加载资源"此种操作方式做解析,其它的方式的过程大同小 ...
- 洛谷11月月赛round.1
太感动了#2 thwfhk 240 (801ms) 100 100 40 又一张明信片,话说10月的怎么还没收到 P2246 SAC#1 - Hello World(升级版) 题目背景 一天, ...
- java1.8常用的函数式接口
//常用函数式接口 final ; //num++; //第一个为传入参数的类型:第二个为返回数据的类型 Function<int[],String> function = (from) ...
- 前端MVC学习总结——AngularJS验证、过滤器
前端MVC学习总结--AngularJS验证.过滤器 目录 一.验证 二.过滤器 2.1.内置过滤器 2.1.1.在模板中使用过滤器 2.1.2.在脚本中调用过滤函数 2.2.自定义过滤器 三.指令( ...
- 选中多个<ul>中的第一个<li>方法
获取第一个ul中的第一个li标签的方法: //获取第一个ul中的第一个li /* $("ul li:first").css("background"," ...
- 【笔记6】用pandas实现条目数据格式的推荐算法 (基于物品的协同)
''' 基于物品的协同推荐 矩阵数据 说明: 1.修正的余弦相似度是一种基于模型的协同过滤算法.我们前面提过,这种算法的优势之 一是扩展性好,对于大数据量而言,运算速度快.占用内存少. 2.用户的评价 ...
- 一张图看懂git push
基本用法 上面的四条命令在工作目录.暂存目录(也叫做索引)和仓库之间复制文件. git add files 把当前文件放入暂存区域. git commit 给暂存区域生成快照并提交. git rese ...
- python实现计算器
计算器功能 实现优先级解析,加减乘除四则运算 自定义小数位精度 实现思想: 先找到最里层括号,根据乘除,加减优先级,调用写好的乘除.加减运算函数算出括号内总值,再将原括号式用所得值替换,此过程循环进行 ...
- PHP7中php.ini、php-fpm和www.conf的配置
引自:https://typecodes.com/web/php7configure.html 1 配置php.ini php.ini是php运行核心配置文件: ######避免PHP信息暴露在htt ...