php异常处理以及laravel框架自定义异常处理

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
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容