首页 > 编程语言 >PHP基础之错误与异常

PHP基础之错误与异常

时间:2024-06-22 17:32:57浏览次数:24  
标签:函数 自定义 错误 error PHP 异常

目录

1 错误

1.1 简介

PHP 中,默认的错误处理很简单。一条错误消息会被发送到浏览器,这条消息带有文件名、行号以及描述错误的消息。
在创建脚本和 Web 应用程序时,错误处理是一个重要的部分。如果代码缺少错误检测编码,那么程序看上去很不专业,也为安全风险敞开了大门。

1.2 简单错误处理

1.2.1 使用die

如下实例展示了一个打开文本文件的简单脚本:

<?php
$file=fopen("welcome.txt","r");
?>

如果文件不存在,会得到类似这样的错误:

Warning: fopen(welcome.txt) [function.fopen]: failed to open stream:
No such file or directory in /www/runoob/test/test.php on line 2

为了避免用户得到类似上面的错误消息,我们在访问文件之前检测该文件是否存在:

<?php
if(!file_exists("welcome.txt"))
{
    die("文件不存在");
}
else
{
    $file=fopen("welcome.txt","r");
}
?>

现在,如果文件不存在,会得到类似这样的错误消息:文件不存在
相比之前的代码,上面的代码更有效,这是由于它采用了一个简单的错误处理机制在错误之后终止了脚本。

1.2.2 die和exit区别

PHP 中,dieexit 是几乎相同的功能,它们都用于终止脚本的执行。两者在功能上没有本质的区别,它们都是语言结构(不是函数),都可以接受一个可选的字符串参数作为退出信息。

  • die()
    die()exit() 的别名。当调用 die() 时,PHP 会输出传递给它的字符串(如果有的话),然后终止脚本的执行。
  • exit()
    exit() 也是用来终止脚本的执行的。可以给它传递一个状态码(一个整数)或一个字符串作为退出信息。如果传递了一个字符串,该字符串会被输出。
php
<?php  
exit("脚本已终止"); // 输出 "脚本已终止" 然后终止  
exit(1); // 终止脚本并返回状态码 1  
?>

注意事项:
如果 exit 或 die 被调用在一个包含有 return 语句的函数中,脚本也会终止,但 return 语句不会被执行。但是,如果 exit 或 die 语句在 return 语句之后,那么 exit 或 die 实际上永远不会被执行,因为函数在 return 语句执行时就已经结束了

使用 exit 或 die 时要小心,因为它们会立即停止脚本的执行,可能会导致数据丢失或其他未预期的副作用。在可能的情况下,最好使用更精细的控制流结构(如 if 语句、break、continue 等)来管理脚本的流程。

1.3 自定义错误处理

1.3.1 定义

创建一个自定义的错误处理器非常简单。创建了一个专用函数,可以在 PHP 中发生错误时调用该函数。
该函数必须有能力处理至少两个参数 (error_level 和 error_message),但是可以接受最多五个参数(可选的:file, line-number 和 error context):

语法:
error_function(error_level,error_message,error_file,error_line,error_context)

参数 描述
error_level 必需。为用户定义的错误规定错误报告级别。必须是一个数字
error_message 必需。为用户定义的错误规定错误消息
error_file 可选。规定错误发生的文件名
error_line 可选。规定错误发生的行号
error_context 可选。规定一个数组,包含了当错误发生时在用的每个变量以及它们的值

错误报告级别
这些错误报告级别是用户自定义的错误处理程序处理的不同类型的错误:

常量 描述
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,但可被用户定义的处理程序捕获
8191 E_ALL 所有错误和警告。(在 PHP 5.4 中,E_STRICT 成为 E_ALL 的一部分)

1.3.2 创建错误函数

创建一个处理错误的函数:

function customError($errno, $errstr)
{
    echo "<b>Error:</b> [$errno] $errstr<br>";
    echo "脚本结束";
    die();
}

上面的代码是一个简单的错误处理函数。当它被触发时,它会取得错误级别和错误消息。然后它会输出错误级别和消息,并终止脚本。
现在,需要确定在何时触发该函数。

PHP 的默认错误处理程序是内建的错误处理程序。打算把上面的函数改造为脚本运行期间的默认错误处理程序。可以修改错误处理程序,使其仅应用到某些错误,这样脚本就能以不同的方式来处理不同的错误。然而,在本例中,打算针对所有错误来使用我们自定义的错误处理程序:

set_error_handler("customError");

由于我们希望我们的自定义函数能处理所有错误,set_error_handler() 仅需要一个参数,可以添加第二个参数来规定错误级别
通过尝试输出不存在的变量,来测试这个错误处理程序:

<?php
// 错误处理函数
function customError($errno, $errstr)
{
    echo "<b>Error:</b> [$errno] $errstr";
}

// 设置错误处理函数
set_error_handler("customError");

// 触发错误
echo($test);
?>

1.4 触发错误

在脚本中用户输入数据的位置,当用户的输入无效时触发错误是很有用的。在 PHP 中,这个任务由 trigger_error() 函数完成。

在本例中,如果 "test" 变量大于 "1",就会发生错误:

<?php
$test=2;
if ($test>1)
{
    trigger_error("变量值必须小于等于 1");
}
?>

以上代码的输出如下所示:

Notice: 变量值必须小于等于 1
in /www/test/test.php on line 5

可以在脚本中任何位置触发错误,通过添加的第二个参数,能够规定所触发的错误类型。

1.5 抑制错误

1.5.1 行内错误抑制

可以让 PHP 利用错误控制操作符 @ 来抑制特定的错误。将这个操作符放置在表达式之前,其后的任何错误都不会出现。

<?php
echo @$foo['bar'];

如果 $foo['bar'] 存在,程序会将结果输出,如果变量 $foo 或是 'bar' 键值不存在,则会返回 null 并且不输出任何东西。如果不使用错误控制操作符,这个表达式会产生一个错误信息 PHP Notice: Undefined variable: foo 或 PHP Notice: Undefined index: bar

注意PHP 处理使用 @ 的表达式比起不用时效率会低一些。
错误控制操作符会 完全 吃掉错误。不但没有显示,而且也不会记录在错误日志中。此外,在正式环境中 PHP 也没有办法关闭错误控制操作符。也许那些错误是无害的,不过那些较具伤害性的错误同时也会被隐藏。

2 异常

2.1 引言

异常是许多流行编程语言的标配,但它们往往被 PHP 开发人员所忽视。像 Ruby 就是一个极度重视异常的语言,无论有什么错误发生,像是 HTTP 请求失败,或者数据库查询有问题,甚至找不到一个图片资源,Ruby (或是所使用的 gems),将会抛出异常,可以通过屏幕立刻知道所发生的问题。

PHP 处理这个问题则比较随意,调用 file_get_contents() 函数通常只会给出 FALSE 值和警告。许多较早的 PHP 框架比如 CodeIgniter 只是返回 false,将信息写入专有的日志,或者使用类似 $this->upload->get_error() 的方法来查看错误原因。这里的问题在于必须找出错误所在,并且通过翻阅文档来查看这个类使用了什么样的错误的方法,而不是明确的暴露错误。

2.2 什么是异常

PHP 5 提供了一种新的面向对象的错误处理方法。
异常处理 用于在指定的错误(异常)情况发生时改变脚本的正常流程。这种情况称为异常。

当异常被触发时,通常会发生:

  • 当前代码状态被保存
  • 代码执行被切换到预定义(自定义)的异常处理器函数
  • 根据情况,处理器也许会从保存的代码状态重新开始执行代码,终止脚本执行,或从代码中另外的位置继续执行脚本

当异常被抛出时,其后的代码不会继续执行,PHP 会尝试查找匹配的 catch 代码块。如果异常没有被捕获,而且又没用使用 set_exception_handler() 作相应的处理的话,那么将发生一个严重的错误(致命错误),并且输出 Uncaught Exception (未捕获异常)的错误消息。

2.3 Try、throw、catch、finally

要避免上面实例中出现的错误,我们需要创建适当的代码来处理异常。

适当的处理异常代码应该包括:

  • Try:使用异常的函数应该位于 "try" 代码块内。如果没有触发异常,则代码将照常继续执行。但是如果异常被触发,会抛出一个异常。
  • Throw:这里规定如何触发异常。每一个 "throw" 必须对应至少一个 "catch"。
  • Catch:捕获异常,并创建一个包含异常信息的对象。
  • finally:用于包含无论是否发生异常都需要执行的代码。这通常用于清理资源,如关闭文件句柄、数据库连接或执行其他必须完成的操作

让我们触发一个异常:

<?php
// 创建一个有异常处理的函数
function checkNum($number)
{
    if($number>1)
    {
        throw new Exception("变量值必须小于等于 1");
    }
        return true;
}    
// 在 try 块 触发异常
try
{
    checkNum(2);
    // 如果抛出异常,以下文本不会输出
    echo '如果输出该内容,说明 $number 变量';
}
// 捕获异常
catch(Exception $e)
{
    echo 'Message: ' .$e->getMessage();
}
?>

上面代码将得到类似这样一个错误:Message: 变量值必须小于等于 1

2.4 自定义异常

创建自定义的异常处理程序非常简单。创建了一个专门的类,当 PHP 中发生异常时,可调用其函数。该类必须是 exception 类的一个扩展。

这个自定义的 customException 类继承了 PHPexception 类的所有属性,可向其添加自定义的函数。

<?php
class customException extends Exception
{
    public function errorMessage()
    {
        // 错误信息
        $errorMsg = '错误行号 '.$this->getLine().' in '.$this->getFile()
        .': <b>'.$this->getMessage().'</b> 不是一个合法的 E-Mail 地址';
        return $errorMsg;
    }
}
$email = "someone@example...com"; 
try
{
    // 检测邮箱
    if(filter_var($email, FILTER_VALIDATE_EMAIL) === FALSE)
    {
        // 如果是个不合法的邮箱地址,抛出异常
        throw new customException($email);
    }
} 
catch (customException $e)
{
//display custom message
echo $e->errorMessage();
}
?>

上面的代码抛出了一个异常,并通过一个自定义的 exception 类来捕获它:

  • customException() 类是作为旧的 exception 类的一个扩展来创建的。这样它就继承了旧的 exception 类的所有属性和方法。
  • 创建 errorMessage() 函数。如果 e-mail 地址不合法,则该函数返回一条错误消息。
  • $email 变量设置为不合法的 e-mail 地址字符串。
  • 执行 "try" 代码块,由于 e-mail 地址不合法,因此抛出一个异常。
  • "catch" 代码块捕获异常,并显示错误消息。

2.5 设置顶层异常处理器

set_exception_handler() 函数可设置处理所有未捕获异常的用户定义函数。

<?php
function myException($exception)
{
    echo "<b>Exception:</b> " , $exception->getMessage();
}
 
set_exception_handler('myException');
 
throw new Exception('Uncaught Exception occurred');
?>

以上代码的输出如下所示:Exception: Uncaught Exception occurred
在上面的代码中,不存在 "catch" 代码块,而是触发顶层的异常处理程序。应该使用此函数来捕获所有未被捕获的异常。

3 错误与异常总结

3.1 区别

PHP 中,错误(Errors)和异常(Exceptions)是两个不同的概念,用于处理不同类型的运行时问题。它们之间的主要区别如下:

  • 错误(Errors
    • 错误通常是更低级别的、更严重的运行时问题,这些问题通常是由于 PHP 引擎内部发生的问题或违反了语言规则导致的。
    • 错误包括如类型错误(Type Errors)、致命错误(Fatal Errors)、解析错误(Parse Errors)等。
    • 错误通常不能被捕获(除了使用特殊的错误处理机制,如 register_shutdown_function() 和 error_get_last()),并且它们会终止脚本的执行。
    • PHP 7 引入了可捕获的错误(Catchable Errors),这些错误可以通过 Error 类进行捕获和处理,但它们在本质上仍然是错误,而不是异常。
  • 异常(Exceptions)
    • 异常是程序执行过程中出现的异常情况,通常是由于程序逻辑错误或不可预见的条件导致的。
    • 异常可以被抛出(throw)和捕获(catch),允许开发者在异常发生时采取适当的措施,如记录日志、回滚事务或执行其他恢复操作。
    • 异常可以通过继承 Exception 类或其子类来定义自定义的异常类型。
    • 当异常被抛出时,程序的执行会立即停止当前的操作,并跳转到最近的匹配的 catch 块(如果存在)。如果没有匹配的 catch 块,异常将冒泡到调用栈的上一层,直到被捕获或到达脚本的顶层并导致脚本终止。

区别

  • 严重性和级别:错误通常表示更严重的、更低级别的问题,而异常通常表示程序逻辑上的错误或异常情况。
  • 处理方式:错误通常不能被直接捕获(除了可捕获的错误),而异常可以通过 try/catch 块进行捕获和处理。
  • 中断执行:错误通常会立即终止脚本的执行,而异常可以通过捕获来避免脚本的终止,并允许开发者执行恢复操作。
  • 自定义性:异常可以通过继承 Exception 类或其子类来定义自定义的异常类型,而错误通常不能自定义(除了用户定义的错误处理器)。

标签:函数,自定义,错误,error,PHP,异常
From: https://www.cnblogs.com/jingzh/p/18262534

相关文章

  • PHP环境搭建之使用PhpStudy
    目录1PhpStudy1.1简介1.2下载&安装1.3修改配置1.3.1Apache配置1.3.2MySQL配置1.3.3MySQL启动问题1.4Composer1.4.1简介1.4.2下载安装1.4.3修改配置1.4.4使用命令1PhpStudy1.1简介phpstudy是一个php运行环境的集成包,用户不需要去配置运行环境,就可以使用,phpstudy不......
  • PHP基础之过滤器讲解
    目录1过滤器1.1简介1.2函数和过滤器1.2.1方法说明1.2.2filter_var示例1.2.3filter_input示例1.2.4filter_var_array和filter_input_array示例1.3自定义过滤器1.4PHPFilter函数1过滤器PHP过滤器用于验证和过滤来自非安全来源的数据,比如用户的输入。1.1简介PHP过......
  • 怎样利用 Clojure 的宏来创建自定义的控制结构,并且如何避免常见的错误?
    在Clojure中,宏是一种宏展开的机制,它可以用于创建自定义的控制结构。通过使用宏,你可以在编写代码时引入新的语法,从而使代码更具可读性和表达力。要创建一个宏,你需要使用defmacro宏,并将宏名称与一个参数列表和一个展开形式绑定。这个展开形式将在宏被调用时用于生成代码。......
  • JetBrains PhpStorm 2024 mac/win版:探索PHP之美,智慧编程新境界
    JetBrainsPhpStorm2024是一款卓越的PHP集成开发环境(IDE),专为满足现代PHP开发者的需求而精心打造。它凭借强大的功能和出色的性能,赢得了全球开发者的广泛赞誉。PhpStorm2024mac/win版获取PhpStorm2024提供了智能的代码编辑功能,包括自动补全、语法高亮、代码重构等,使得编写......
  • php8开启jit
    JIT时php8的重要功能之一,可以极大的提高性能;JIT编译器集成在了Opcache插件中,仅在启动Opcache插件才有效Opcache将PHP脚本编译后的字节码存储到内存中,以避免每次执行脚本时重新解析和编译,从而提高PHP应用程序的性能和响应速度。 Opcache开启:编辑php.ini文件,将zend_exten......
  • 深入了解Java异常处理
    在Java编程中,异常处理是一个至关重要的概念。通过正确地处理异常,程序员可以编写出健壮且易于维护的代码,提升程序的可靠性。本文将详细介绍Java的异常处理机制,包括异常的分类、捕获和处理异常的语法、常见的异常类型以及自定义异常的实现。一、什么是异常异常是程序运行过......
  • php+sql后台实现从主表迁出至副表(数据超万条)
    上万条甚至上百万数据进行迁出做备份或者进行不妨碍原系统数据的操作,现在很多企业都会用到,目前就需要将上百万条数据进行迁出到副表保存并操作,直接再后台写一个按钮进行操作,既方便操作也不会很慢。毕竟是客户需要,不能每次迁出都要客户去数据库操作,操作的不好那数据危险度挺高的。......
  • Ubuntu 24.04 编译 PHP 8.0
    PKG_CONFIG_PATH=/usr/local/openssl111w/lib/pkgconfigOPENSSL_CONF=/usr/local/openssl111w/ssl/openssl.cnf./configure--prefix=/usr/local/php80--enable-fpm--with-fpm-user=www-data--with-fpm-group=www-data--enable-phpdbg--disable-short-tags--wit......
  • Javase.认识异常
    认识异常【本章目标】1.异常的概念与体系结构1.1异常的概念1.2异常的体系结构1.3异常的分类2.异常的处理2.1防御式编程2.2异常的抛出2.3异常的捕获2.3.2try-catch捕获并处理2.3.3finally2.4异常的处理流程3.自定义异常类【本章目标】异常概念与体系结......
  • 异常值检测(1)——箱线图四分位距和3σ
            说白了,异常值就是那些偏离多数样本值过多的值,比如我用机器学习在做房价预测时,获取的沈阳浑南区数据普遍都在11000左右,结果有那么一两个楼盘是20000(管他是不是碧桂园),那么这两个值就是异常值。        异常值会严重干扰模型的性能,包括降低预测能力、......