一些簡單的錯(cuò)誤處理函數(shù)(二)接下來,我們繼續(xù)學(xué)習(xí) PHP 中的錯(cuò)誤處理函數(shù)。上次學(xué)習(xí)過的函數(shù)是錯(cuò)誤信息的獲取、設(shè)置、發(fā)送等功能,今天學(xué)習(xí)的內(nèi)容主要是關(guān)于錯(cuò)誤的捕獲相關(guān)的函數(shù)。 set_error_handler()首先是大家可能會(huì)接觸過的一個(gè)函數(shù),它可以用來捕獲一些錯(cuò)誤的信息。如果我們需要統(tǒng)一處理一些錯(cuò)誤,比如規(guī)定日志格式或者將錯(cuò)誤信息發(fā)送到郵件中,一般會(huì)在入口文件的開頭在全局范圍內(nèi)定義一個(gè)這個(gè)函數(shù)進(jìn)行統(tǒng)一的處理。 echo $a; // Notice: Undefined variable: a ... set_error_handler() 函數(shù)接收一個(gè)回調(diào)函數(shù)和一個(gè)錯(cuò)誤接收的類型,它的函數(shù)簽名是: set_error_handler ( callable $error_handler [, int $error_types = E_ALL | E_STRICT ] ) : mixed $error_handler是一個(gè)回調(diào)(匿名)函數(shù),這個(gè)函數(shù)內(nèi)部可以獲取到錯(cuò)誤的等級(jí)、信息、文件、行數(shù)等 handler ( int $errno , string $errstr [, string $errfile [, int $errline [, array $errcontext ]]] ) : bool 其中,$errcontext 已經(jīng)在 PHP7.2 之后取消了。 \$error_types,用于錯(cuò)誤接收的類型,就像 error_reporting() 函數(shù)定義的錯(cuò)誤類型一樣,它用于控制 $error_handler 回調(diào)函數(shù)所能接收的錯(cuò)誤的類型。 需要注意的是,這個(gè)函數(shù)無法處理 E_ERROR、 E_PARSE、 E_CORE_ERROR、 E_CORE_WARNING、 E_COMPILE_ERROR、 E_COMPILE_WARNING 這些類型的錯(cuò)誤,也就是說,能夠中斷程序執(zhí)行的錯(cuò)誤它都無法捕獲處理。 另外,在函數(shù)調(diào)用之前的錯(cuò)誤是無法捕獲到的,只有在函數(shù)調(diào)用之后的錯(cuò)誤才能通過這個(gè)函數(shù)進(jìn)行捕獲處理。 restore_error_handler()這個(gè)函數(shù)是用于還原之前的錯(cuò)誤處理函數(shù)。比如我們?cè)谏厦娲a下添加這個(gè)函數(shù),然后再次觸發(fā)錯(cuò)誤,錯(cuò)誤將會(huì)使用回 PHP 的標(biāo)準(zhǔn)處理程序。 restore_error_handler(); set_exception_handler()學(xué)習(xí)了上面錯(cuò)誤處理的函數(shù)后,從名稱就可以看出,這個(gè)函數(shù)是用來處理異常的,它可以在全局范圍內(nèi)捕獲異常。 set_exception_handler(function($ex){ 它的函數(shù)簽名是: set_exception_handler ( callable $exception_handler ) : callable 只接收一個(gè)回調(diào)函數(shù),回調(diào)函數(shù)中只有一個(gè)參數(shù),是一個(gè) Exception 類型的參數(shù)內(nèi)容,就和 try...catch 中的 catch 塊的參數(shù)一樣。在 PHP7 以后接收到的是一個(gè) Throwable 類型的參數(shù)。也就是說,它可以捕獲到所有的錯(cuò)誤和異常。 不過需要注意的是,在 PHP 中,所有的異常如果不進(jìn)行處理,都會(huì)以中止腳本的錯(cuò)誤形式返回報(bào)錯(cuò)信息。所以,在 set_exception_handler() 內(nèi)處理完之后,腳本會(huì)中止運(yùn)行。即使后面還有代碼。所以,這個(gè)函數(shù)一般會(huì)用于全局捕獲一些異常、錯(cuò)誤后進(jìn)行日志記錄,它不具有 try...catch 的能力,讓異常處理完成后還能繼續(xù)進(jìn)行其他操作。 restore_exception_handler()同樣的,異常捕獲也是可以進(jìn)行回退的。 set_exception_handler(function($ex){ 我們定義了兩個(gè) set_exception_handler() 函數(shù),當(dāng)使用 restore_exception_handler() 后,拋出的異常將會(huì)進(jìn)入到第一個(gè) set_exception_handler() 函數(shù)中進(jìn)行處理。同理,restore_error_handler() 函數(shù)如果定義了多個(gè)錯(cuò)誤處理,使用 restore_error_handler() 后也會(huì)一級(jí)一級(jí)回退,直到最終使用 PHP 的錯(cuò)誤處理流程進(jìn)行處理。 trigger_error()最后,我們來看看如何手動(dòng)拋出一個(gè)錯(cuò)誤。就像上面例子中的 throw new Exception() 一樣,PHP 也提供了一個(gè)用戶自定義手動(dòng)拋出錯(cuò)誤的函數(shù)。 trigger_error("I'm Error One!"); // Notice: I'm Error One! 它的函數(shù)簽名是: trigger_error ( string $error_msg [, int $error_type = E_USER_NOTICE ] ) : bool $error_msg,也就是這個(gè)錯(cuò)誤的具體信息,長度限制為 1024 個(gè)字節(jié),如果超過了這個(gè)長度就會(huì)被截?cái)唷A硗?,如果這個(gè)信息中包含 HTML 實(shí)體標(biāo)簽的話,也不會(huì)直接轉(zhuǎn)義,在網(wǎng)頁顯示時(shí)需要使用 htmlentities() 來進(jìn)行處理。 $error_type參數(shù)則是指定報(bào)錯(cuò)的級(jí)別,默認(rèn)是 E_USER_NOTICE ,而且它只支持 E_USER... 相關(guān)的錯(cuò)誤信息。也就是說,它的參數(shù)只能填三個(gè) E_USER_NOTICE 、 E_USER_WARNING 、 E_USER_ERROR 。 當(dāng)然,我們手動(dòng)拋出的錯(cuò)誤信息也是可以通過 set_error_handler() 進(jìn)行捕獲的。 set_error_handler(function($errno, $errstr, $errfile, $errline){ 最后一個(gè) trigger_error() 我們使用了 E_WARNING 類型,可以看出直接返回的內(nèi)容是 指定的錯(cuò)誤類型無效 ,而不是我們定義的內(nèi)容。也就是說,這里是這個(gè)函數(shù)的參數(shù)類型錯(cuò)誤的報(bào)錯(cuò),不是我們手動(dòng)想拋出的錯(cuò)誤了。 總結(jié)其實(shí) PHP 的錯(cuò)誤處理函數(shù)也就這些了,在 PHP7 下面,大部分錯(cuò)誤都可以通過異常捕獲了,也就是說,PHP 越向后發(fā)展越會(huì)通過面向?qū)ο蟮姆绞絹硖幚磉@些錯(cuò)誤信息。不過,我們依然還是要對(duì)他們有全面的了解,畢竟在 PHP 的版本更新中,短時(shí)間還不會(huì)完全的摒棄錯(cuò)誤處理的場景,在之后我們學(xué)習(xí)詳細(xì)的異常處理相關(guān)的知識(shí)時(shí),說不定還會(huì)再次見到它們的身影。 測試代碼: https://github.com/zhangyue0503/dev-blog/blob/master/php/202004/source/%E4%B8%80%E4%BA%9B%E7%AE%80%E5%8D%95%E7%9A%84%E9%94%99%E8%AF%AF%E5%A4%84%E7%90%86%E5%87%BD%E6%95%B0%EF%BC%88%E4%BA%8C%EF%BC%89.php 參考文檔: https://www./manual/zh/function.set-error-handler.phphttps://www./manual/zh/function.set-exception-handler.phphttps://www./manual/zh/function.restore-error-handler.phphttps://www./manual/zh/function.restore-exception-handler.php |
|