php中应通过errno和error_reporting级别区分用户错误与系统错误:用户错误由业务逻辑拦截,系统错误触发e_warning及以上级别;用set_error_handler()分流处理,e_user_*归用户错误不记日志,原生错误按关键路径判定为系统错误并告警。

PHP 中怎么判断是用户输错了还是服务器崩了
直接看错误的 errno 和 error_reporting() 级别:用户错误(比如表单填错、参数非法)该由业务逻辑拦截,不该触发 PHP 错误;系统错误(如 fopen() 失败、数据库连接中断)才会抛出 E_WARNING 或更高级别错误。混着处理,日志就乱,报警也失真。
用 set_error_handler() 区分两类错误的典型写法
默认错误处理会把所有错误都当一回事,但你可以自己接管,按类型分流:
- 对
E_USER_*类错误(trigger_error('用户名为空', E_USER_WARNING)),一律归为用户错误,不记录到 error log,只返回给前端提示 - 对
E_WARNING、E_ERROR、E_PARSE等原生错误,检查是否发生在关键路径(如数据库操作、文件读写),是则打标为系统错误,发 Slack 告警 + 记入sys_error.log - 注意:不要在 handler 里调用
error_log()写文件,高并发下容易阻塞;改用syslog()或异步队列 -
error_get_last()在 shutdown 阶段不可靠,别依赖它补全系统错误上下文
为什么 try/catch 捕不到 fopen() 失败这类错误
因为 fopen() 默认不抛异常,只发警告(E_WARNING)。想用 try/catch 统一处理,得提前转换:
采用 php+mysql 数据库方式运行的强大网上商店系统,执行效率高速度快,支持多语言,模板和代码分离,轻松创建属于自己的个性化用户界面 v3.5更新: 1).进一步静态化了活动商品. 2).提供了一些重要UFT-8转换文件 3).修复了除了网银在线支付其它支付显示错误的问题. 4).修改了LOGO广告管理,增加LOGO链接后主页LOGO路径错误的问题 5).修改了公告无法发布的问题,可能是打压
- 启用
set_error_handler()并在其中对E_WARNING调用throw new ErrorException()—— 但仅限开发环境,线上慎用,性能损耗明显 - 更稳妥的做法:所有 I/O 操作手动包装,比如
safe_file_get_contents($path),内部先@fopen()抑制警告,再根据返回值决定抛InvalidArgumentException(用户错)还是RuntimeException(系统错) - 别信
file_exists()的返回值,它和后续fopen()之间存在竞态;判断依据必须是操作本身是否成功
日志里怎么一眼看出哪个是用户错、哪个是系统错
靠字段,不靠文字描述。在写日志前统一加一个 error_category 字段:
立即学习“PHP免费学习笔记(深入)”;
- 用户错误:固定写
"category": "user",附带user_id、http_status: 400、trace_id - 系统错误:固定写
"category": "system",必须含errno、file、line、backtrace(裁剪掉 vendor 内部帧) - 别用日志级别(INFO/WARN/ERROR)区分——很多团队把用户参数校验失败也记成 ERROR,结果告警风暴
- NGINX access log 里加
$upstream_http_x_error_category,方便和 PHP 日志关联排查
真正难的不是分两类错误,而是让所有协作方(前端、测试、运维)对“什么算用户错”有共识。比如「支付接口返回余额不足」是用户错,但「余额查询时 Redis 连接超时」就是系统错——这个边界一旦模糊,监控指标就失效。










