PHP异常处理核心是try-catch结构,仅捕获Exception及其子类异常(如PDOException),不捕获致命错误、警告或通知;需按子类到父类顺序写catch,避免空catch,善用finally做清理,推荐自定义语义化异常类。

PHP异常处理的核心是try-catch结构,它让程序在出错时能优雅降级、避免崩溃,并提供调试线索。关键不在于“捕获所有错误”,而在于只捕获可预期、可恢复的异常,把真正的问题暴露出来。
哪些错误能被 try-catch 捕获?
try-catch 只捕获Exception(及继承它的类)抛出的异常,比如手动 throw new Exception(),或系统抛出的 PDOException、InvalidArgumentException 等。
它不能捕获 PHP 致命错误(如语法错误、未定义函数、内存耗尽)、警告(Warning)或通知(Notice)。这类问题需靠 error_reporting、set_error_handler 或 register_shutdown_function 配合处理。
基础 try-catch 写法与常见误区
最简结构如下:
php
try {
// 可能出异常的代码,如数据库查询、文件读取
$pdo->query("SELECT * FROM users");
} catch (PDOException $e) {
// 精确捕获 PDO 相关异常
error_log("DB Error: " . $e->getMessage());
echo "数据加载失败,请稍后重试";
} catch (Exception $e) {
// 保底捕获其他异常(放最后)
error_log("Unexpected error: " . $e->getMessage());
}
?>
立即学习“PHP免费学习笔记(深入)”;
- 多个 catch 块按子类在前、父类在后排列,否则子类永远无法命中
- 不要空 catch —— 即使只是记录日志,也别直接吞掉异常
- 避免用 Exception 通吃一切;优先捕获具体异常类型,便于分场景处理
finally:无论成败都执行的清理逻辑
finally 块总会运行,适合释放资源、关闭连接、重置状态等操作:
$file = null;
try {
$file = fopen("data.txt", "r");
if (!$file) throw new RuntimeException("无法打开文件");
// 处理文件内容...
} catch (RuntimeException $e) {
echo "文件操作失败";
} finally {
if ($file && is_resource($file)) {
fclose($file); // 确保文件句柄被关闭
}
}
?>
- 即使 try 中 return 了,finally 仍会先执行
- finally 不用于替代 catch——它不接收异常对象,也不影响异常传播
自定义异常与业务逻辑解耦
为不同业务场景定义专属异常类,能让错误分类更清晰、处理更有针对性:
class InsufficientBalanceException extends Exception {}
class InvalidOrderException extends Exception {}
function processPayment($amount, $balance) {
if ($amount > $balance) {
throw new InsufficientBalanceException("余额不足:{$balance} }
// 其他逻辑...
}
try {
processPayment(1000, 500);
} catch (InsufficientBalanceException $e) {
showTopUpPrompt(); // 引导充值
} catch (InvalidOrderException $e) {
logAndCancelOrder(); // 订单作废并记录
}
?>
- 继承 Exception 即可,无需额外方法;重点是命名体现语义
- 构造时传入有意义的 message,方便日志追踪和前端提示
- 避免把异常当流程控制手段(例如用 throw 来跳转页面),应保持逻辑清晰
基本上就这些。异常处理不是堆砌 try-catch,而是建立一套可读、可维护、有边界的错误响应机制。











