PHP错误:是属于php程序自身的问题,一般是由非法的语法,环境问题导致的,使得编译器无法通过检查,甚至无法运行的情况。平时遇到的warming、notice都是错误,只是级别不同而已。
Fatal Error:致命错误(脚本终止运行)
E_ERROR // 致命的运行错误,错误无法恢复,暂停执行脚本
E_CORE_ERROR // PHP启动时初始化过程中的致命错误
E_COMPILE_ERROR // 编译时致命性错,就像由Zend脚本引擎生成了一个E_ERROR
E_USER_ERROR // 自定义错误消息。像用PHP函数trigger_error(错误类型设置为:E_USER_ERROR)
Parse Error:编译时解析错误,语法错误(脚本终止运行)
E_PARSE //编译时的语法解析错误
Warning Error:警告错误(仅给出提示信息,脚本不终止运行)
E_WARNING // 运行时警告 (非致命错误)。
E_CORE_WARNING // PHP初始化启动过程中发生的警告 (非致命错误) 。
E_COMPILE_WARNING // 编译警告
E_USER_WARNING // 用户产生的警告信息
Notice Error:通知错误(仅给出通知信息,脚本不终止运行)
E_NOTICE // 运行时通知。表示脚本遇到可能会表现为错误的情况.
E_USER_NOTICE // 用户产生的通知信息。
PHP异常:一般是业务逻辑上出现的不合预期、与正常流程不同的状况,不是语法错误。
异常处理用于在指定的错误(异常)情况发生时改变脚本的正常流程,我们可以使用set_error_handler函数设置用户自定义的错误处理函数来处理错误,set_exception_handler函数设置用户自定义的异常处理函数来处理异常。例如我们想把异常和错误都处理成json格式输出和调试,代码如下:
function ErrorHandler($error_level,$error_message,
$error_file,$error_line,$error_request)
{
$httpVersion = "HTTP/1.1";
$contentType = 'application/json';
header($httpVersion. " ". 500 ." ". $error_message);
header("Content-Type:". $contentType);
echo json_encode(['message'=>$error_message,'file'=>$error_file,'line'=>$error_line]);die();
}
function ExceptionHandler($exception)
{
$httpVersion = "HTTP/1.1";
$contentType = 'application/json';
header($httpVersion. " ". $exception->getCode() ." ". "Bad Request");
header("Content-Type:". $contentType);
echo json_encode(['message'=>$exception->getMessage(),'file'=>$exception->getFile(),'line'=>$exception->getLine()]);die();
}
set_error_handler("ErrorHandler");
set_exception_handler('ExceptionHandler');
$a = 1/0;//测试错误
throw new Exception('Uncaught Exception');//测试异常
有时候我们用laravel写api接口的时候代码不严谨导致报错,接口直接返回一个渲染地十分漂亮的html网页,这时候前端就无法解析了,app甚至可能直接崩溃,这样并不好,我们希望的是不管发生什么情况都返回统一格式的json,好统一处理。
接下来讲如何应用到laravel框架中去,最先想到的是写一个中间件,把set_error_handler("ErrorHandler")和
set_exception_handler('ExceptionHandler')方法放进去,然后把上述两个自定义方法全局加载。然后只要用到这个中间件的路由就可以像上述那样将错误和异常处理成统一的json格式输出。但是却发现除了错误能被捕获,其他的都无法捕获。于是我决定去App\Exceptions\Handler类动下手脚。laravel所有异常和错误都由类 App\Exceptions\Handler 处理,该类包含两个方法:report 和 render,report 方法只是将异常传递给异常被记录的基类,我们可以在这个方法下面自定义我们的错误处理方法。
class ExceptionHandler
{
static $httpVersion = "HTTP/1.1";
static $contentType = 'application/json';
public static function getHttpStatusMessage($statusCode){
$httpStatus = array(
100 => 'Continue',
101 => 'Switching Protocols',
200 => 'OK',
201 => 'Created',
202 => 'Accepted',
203 => 'Non-Authoritative Information',
204 => 'No Content',
205 => 'Reset Content',
206 => 'Partial Content',
300 => 'Multiple Choices',
301 => 'Moved Permanently',
302 => 'Found',
303 => 'See Other',
304 => 'Not Modified',
305 => 'Use Proxy',
306 => '(Unused)',
307 => 'Temporary Redirect',
400 => 'Bad Request',
401 => 'Unauthorized',
402 => 'Payment Required',
403 => 'Forbidden',
404 => 'Not Found',
405 => 'Method Not Allowed',
406 => 'Not Acceptable',
407 => 'Proxy Authentication Required',
408 => 'Request Timeout',
409 => 'Conflict',
410 => 'Gone',
411 => 'Length Required',
412 => 'Precondition Failed',
413 => 'Request Entity Too Large',
414 => 'Request-URI Too Long',
415 => 'Unsupported Media Type',
416 => 'Requested Range Not Satisfiable',
417 => 'Expectation Failed',
500 => 'Internal Server Error',
501 => 'Not Implemented',
502 => 'Bad Gateway',
503 => 'Service Unavailable',
504 => 'Gateway Timeout',
505 => 'HTTP Version Not Supported');
return ($httpStatus[$statusCode]) ? $httpStatus[$statusCode] : $httpStatus[500];
}
public static function exception_handler(\Exception $exception){
header("Content-Type:". self::$contentType);
$statusCode = method_exists($exception,'getStatusCode')?$exception->getStatusCode():$exception->getCode();
header(self::$httpVersion. " ". ($statusCode = $statusCode == 0? 500 : $statusCode) ." " . self::getHttpStatusMessage($statusCode));
$response = array(
'data'=>[],
'status_code'=>$statusCode == 0?500:$statusCode,
'error_msg'=>$exception->getMessage(),
'error_code'=>method_exists($exception,'getErrorId')?$exception->getErrorId():self::getHttpStatusMessage($statusCode),
);
if (env('APP_DEBUG')){
$response = array_merge($response,array('debug'=>array(
'file'=>$exception->getFile(),
'line'=>$exception->getLine(),
'trace'=>$exception->getTrace(),
'type'=>get_class($exception)
)));
}
echo json_encode($response);die();
}
}
然后在report方法引用
public function report(Exception $exception)
{
if (PHP_SAPI == 'cli'){ //判断是不是cli模式运行
parent::report($exception);
}else{
\Linyuee\ExceptionHandler::exception_handler($exception);
}
}
这样异常和错误都能捕获处理成json格式了
推介我写的一个包,直接comp安装然后只要像上述那样在report方法引入就能用了。
composer require linyuee/laravel-exception