1、異常概述
異常(Exception)是一種錯(cuò)誤處理機(jī)制,用于在指定的錯(cuò)誤發(fā)生時(shí)改變腳本的正常流程。
當(dāng)異常被觸發(fā)時(shí),當(dāng)前代碼狀態(tài)被保存,代碼執(zhí)行被切換到預(yù)定義的異常處理器函數(shù)(如果有)
根據(jù)情況,處理器也許會(huì)從保存的代碼狀態(tài)重新開(kāi)始執(zhí)行代碼,終止腳本執(zhí)行,或從代碼中另外的位置繼續(xù)執(zhí)行腳本
2、異常的基本使用
當(dāng)異常被拋出時(shí),其后的代碼不會(huì)繼續(xù)執(zhí)行,PHP 會(huì)嘗試查找匹配的 "catch" 代碼塊。
如果異常沒(méi)有被捕獲,而且又沒(méi)用使用 set_exception_handler() 作相應(yīng)的處理的話,那么將發(fā)生一個(gè)嚴(yán)重的錯(cuò)誤(致命錯(cuò)誤),并且輸出 "Uncaught Exception" (未捕獲異常)的錯(cuò)誤消息。
嘗試拋出一個(gè)異常,同時(shí)不去捕獲它
1
2
3
4
5
6
7
8
|
//create function with an exception function checkNum( $number ){ if ( $number >1){ throw new Exception( "Value must be 1 or below" ); } } //trigger exception checkNum(2); |
上面的代碼會(huì)獲得類似這樣的一個(gè)錯(cuò)誤:
Fatal error: Uncaught exception 'Exception' with message 'Value must be 1 or below' in C:\webfolder\test.php:6
Stack trace: #0 C:\webfolder\test.php(12):checkNum(28) #1 {main} thrown in C:\webfolder\test.php on line 6
注意:PHP默認(rèn)是警告模式,如果需要對(duì)系統(tǒng)錯(cuò)誤使用異常處理機(jī)制,則要在處理代碼之前設(shè)置錯(cuò)誤處理模式
1
2
3
|
set_error_handler( function (){ throw new Exception( '錯(cuò)誤!' ); }); |
3、Try, throw 和 catch
正確的處理程序應(yīng)當(dāng)包括:
Try - 使用異常的函數(shù)應(yīng)該位于 "try" 代碼塊內(nèi)。如果沒(méi)有觸發(fā)異常,則代碼將照常繼續(xù)執(zhí)行。但是如果異常被觸發(fā),會(huì)拋出一個(gè)異常。
Throw - 這里規(guī)定如何觸發(fā)異常。每一個(gè) "throw" 必須對(duì)應(yīng)至少一個(gè) "catch"
Catch - "catch" 代碼塊會(huì)捕獲異常,并創(chuàng)建一個(gè)包含異常信息的對(duì)象
1
2
3
4
5
6
7
8
9
10
11
12
13
|
//創(chuàng)建可拋出一個(gè)異常的函數(shù) function checkNum( $number ){ if ( $number >1){ throw new Exception( "Value must be 1 or below" ); } } //在 "try" 代碼塊中觸發(fā)異常 try { checkNum(2); } catch (Exception $e ) { //捕獲異常 echo 'File: ' . $e ->getFile(). ' line: ' . $e ->getLine(). '<br>' ; die ( 'Message: ' . $e ->getMessage()); } |
上面代碼將獲得類似這樣一個(gè)錯(cuò)誤:
File: E:\webdev\www\pdo\3.php line: 7
Message: Value must be 1 or below
代碼解析:創(chuàng)建 checkNum() 函數(shù)。它檢測(cè)數(shù)字是否大于 1。如果是,則拋出一個(gè)異常。在 "try" 代碼塊中調(diào)用 checkNum(),函數(shù)checkNum() 函數(shù)中的異常被拋出,"catch" 代碼塊接收到該異常,并創(chuàng)建一個(gè)包含異常信息的對(duì)象 ($e)。通過(guò)從這個(gè) exception 對(duì)象輸出來(lái)自該異常的錯(cuò)誤消息
不過(guò),為了遵循“每個(gè) throw 必須對(duì)應(yīng)一個(gè) catch”的原則,可以設(shè)置一個(gè)頂層的異常處理器來(lái)處理漏掉的錯(cuò)誤,見(jiàn)第7點(diǎn)
4、創(chuàng)建自定義的異常處理器
創(chuàng)建一個(gè)專門的類,當(dāng) PHP 中發(fā)生異常時(shí),可調(diào)用其函數(shù)。該類必須是 exception 類的一個(gè)擴(kuò)展。這個(gè)自定義的exception 類繼承了 PHP 的 exception 類的所有屬性,可向其添加自定義的函數(shù)。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
class customException extends Exception{ public function errorMessage(){ return 'Error on line ' . $this ->getLine(). ' in ' . $this ->getFile(). ': <b>' . $this ->getMessage(). '</b> is not a valid E-Mail address' ; } } try { //使用PHP過(guò)濾器驗(yàn)證郵箱有效性 if (filter_var( $email , FILTER_VALIDATE_EMAIL) === FALSE){ throw new customException( $email ); } } catch (customException $e ){ echo $e ->errorMessage(); } |
這個(gè)新的類是舊的 exception 類的副本,外加 errorMessage() 函數(shù)。正因?yàn)樗桥f類的副本,因此它從舊類繼承了屬性和方法,我們可以使用 exception 類的方法,比如 getLine() 、 getFile() 以及 getMessage()。
上面的代碼拋出了一個(gè)異常,并通過(guò)一個(gè)自定義的 exception 類來(lái)捕獲它:創(chuàng)建 errorMessage() 函數(shù)。如果 e-mail 地址不合法,則該函數(shù)返回一條錯(cuò)誤消息
5、多個(gè)異常的捕獲
可以為一段腳本使用多個(gè)異常,來(lái)檢測(cè)多種情況。
可以使用多個(gè) if..else 代碼塊,或一個(gè)switch 代碼塊,或者嵌套多個(gè)異常。這些異常能夠使用不同的 exception 類,并返回不同的錯(cuò)誤消息:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
class customException extends Exception{ public function errorMessage(){ return = 'Error on line ' . $this ->getLine(). ' in ' . $this ->getFile(). ': <b>' . $this ->getMessage(). '</b> is not a valid E-Mail address' ; } } try { if (filter_var( $email , FILTER_VALIDATE_EMAIL) === FALSE){ throw new customException( $email ); } //check for "example" in mail address if ( strpos ( $email , "example" ) !== FALSE){ throw new Exception( "$email is an example e-mail" ); } } catch (customException $e ){ echo $e ->errorMessage(); } catch (Exception $e ){ echo $e ->getMessage(); } |
上面的代碼測(cè)試了兩種條件,如何任何條件不成立,則拋出一個(gè)異常:
執(zhí)行 "try" 代碼塊,在第一個(gè)條件下,不會(huì)拋出異常。由于 e-mail 含有字符串 "example",第二個(gè)條件會(huì)觸發(fā)異常。"catch" 代碼塊會(huì)捕獲異常,并顯示恰當(dāng)?shù)腻e(cuò)誤消息,如果沒(méi)有捕獲 customException,只捕獲了 base exception,則在那里處理異常。
6、重新拋出異常
有時(shí),當(dāng)異常被拋出時(shí),也許希望以不同于標(biāo)準(zhǔn)的方式對(duì)它進(jìn)行處理。可以在一個(gè) “catch” 代碼塊中再次拋出異常。腳本應(yīng)該對(duì)用戶隱藏系統(tǒng)錯(cuò)誤。對(duì)程序員來(lái)說(shuō),系統(tǒng)錯(cuò)誤也許很重要,但是用戶對(duì)它們并不感興趣。為了讓用戶更容易使用,可以再次拋出帶有對(duì)用戶比較友好的消息的異常:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
class customException extends Exception{ public function errorMessage(){ return $this ->getMessage(). ' is not a valid E-Mail address.' ; } } try { try { if ( strpos ( $email , "example" ) !== FALSE){ throw new Exception( $email ); } } catch (Exception $e ){ //re-throw exception throw new customException( $email ); } } catch (customException $e ){ //display custom message echo $e ->errorMessage(); } |
上面的代碼檢測(cè)在郵件地址中是否含有字符串 "example"。如果有,則再次拋出異常:
把 $email 變量設(shè)置為一個(gè)有效的郵件地址,但含有字符串 "example"。"try" 代碼塊包含另一個(gè) "try" 代碼塊,這樣就可以再次拋出異常。由于 e-mail 包含字符串 "example",因此觸發(fā)異常。"catch" 捕獲到該異常,并重新拋出 "customException"。捕獲到 "customException",并顯示一條錯(cuò)誤消息。
如果在其目前的 "try" 代碼塊中異常沒(méi)有被捕獲,則它將在更高層級(jí)上查找 catch 代碼塊。
7、設(shè)置頂層異常處理器
set_exception_handler() 函數(shù)可設(shè)置處理所有未捕獲異常的用戶定義函數(shù)
1
2
3
4
5
|
function myException( $exception ){ echo "<b>Exception:</b> " , $exception ->getMessage(); } set_exception_handler( 'myException' ); throw new Exception( 'Uncaught Exception occurred' ); |
以上代碼的輸出應(yīng)該類似這樣:
Exception: Uncaught Exception occurred
在上面的代碼中,不存在 "catch" 代碼塊,而是觸發(fā)頂層的異常處理程序。應(yīng)該使用此函數(shù)來(lái)捕獲所有未被捕獲的異常。
但是對(duì)于系統(tǒng)自動(dòng)拋出的錯(cuò)誤,會(huì)先經(jīng)過(guò)set_error_handler處理并拋出異常才能被set_exception_handler處理
1
2
3
4
5
6
7
8
|
function myException( $exception ){ echo "<b>Exception:</b> " , $exception ->getMessage(); } set_exception_handler( 'myException' ); set_error_handler( function (){ throw new Exception( '錯(cuò)誤!' ); }); echo 10/0; //觸發(fā)被除數(shù)不能為0的警告 |
代碼執(zhí)行結(jié)果:Exception: 錯(cuò)誤!
總結(jié)
到此這篇關(guān)于PHP中異常處理機(jī)制的文章就介紹到這了,更多相關(guān)PHP異常處理機(jī)制內(nèi)容請(qǐng)搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!
原文鏈接:https://juejin.im/post/6893305918243471373