PHP錯(cuò)誤處理&異常處理方式小結(jié)
1、錯(cuò)誤處理的分類(lèi)
? 語(yǔ)法錯(cuò)誤: 違背了PHP程序語(yǔ)言的規(guī)則錯(cuò)誤,稱之為語(yǔ)法錯(cuò)誤。
? 運(yùn)行錯(cuò)誤: 經(jīng)過(guò)語(yǔ)法錯(cuò)誤檢測(cè)后,將開(kāi)始運(yùn)行 PHP 代碼,在此發(fā)生的錯(cuò)誤為運(yùn)行時(shí)錯(cuò)誤。
? 邏輯錯(cuò)誤: 邏輯錯(cuò)誤是指程序開(kāi)發(fā)過(guò)程中由于業(yè)務(wù)邏輯造成錯(cuò)誤。
print_r(數(shù)組類(lèi)型) //沒(méi)有以分號(hào)結(jié)束語(yǔ)句,報(bào)語(yǔ)法錯(cuò)誤 /* 加載不存在文件 連接數(shù)據(jù)庫(kù)失敗 遠(yuǎn)程請(qǐng)求失敗 函數(shù)或類(lèi)不存在 */ require 'request.php'; //因?yàn)榧虞d不存在文件,所以會(huì)發(fā)生運(yùn)行時(shí)錯(cuò)誤 echo (10/0); //邏輯錯(cuò)誤,這IDE都會(huì)檢測(cè)到,一般都有提示 //邏輯錯(cuò)誤 for ($i = 0; $i < 5; $i--) { echo $i; }
2、常見(jiàn)錯(cuò)誤報(bào)告類(lèi)型
值 | 常量 | 描述 |
---|---|---|
1 | E_ERROR | 致命的運(yùn)行時(shí)錯(cuò)誤。這類(lèi)錯(cuò)誤一般是不可恢復(fù)的情況,例如內(nèi)存分配導(dǎo)致的問(wèn)題。后果是導(dǎo)致腳本終止不再繼續(xù)運(yùn)行 |
2 | E_WARNING | 運(yùn)行時(shí)警告 (非致命錯(cuò)誤)。僅給出提示信息,但是腳本不會(huì)終止運(yùn)行。 |
8 | E_NOTICE | 運(yùn)行時(shí)通知。表示腳本遇到可能會(huì)表現(xiàn)為錯(cuò)誤的情況。 |
64 | E_COMPILE_ERROR | 致命編譯時(shí)錯(cuò)誤。類(lèi)似 E_ERROR |
2048 | E_STRICT | 啟用 PHP 對(duì)代碼的修改建議,以確保代碼具有最佳的互操作性和向前兼容性。 |
8192 | E_DEPRECATED | 運(yùn)行時(shí)通知。啟用后將會(huì)對(duì)在未來(lái)版本中可能無(wú)法正常工作的代碼給出警告。 |
8191 | E_ALL | 所有錯(cuò)誤和警告,除級(jí)別 E_STRICT 以外。 |
注意:修改 php.ini
配置文件, display_errors = On
即開(kāi)啟或者關(guān)閉錯(cuò)誤顯示。
關(guān)閉警告與致命錯(cuò)誤
error_reporting(~E_WARNING & ~E_COMPILE_ERROR); require('a');
顯示除通知外的所有錯(cuò)誤
error_reporting(E_ALL & ~E_NOTICE); echo $arrays;
關(guān)閉錯(cuò)誤顯示
error_reporting(0);
3、異常處理
基本語(yǔ)法:
try{ //邏輯代碼 }catch(){ //捕獲異常時(shí)候的處理 }
注意: 需要手動(dòng)拋出異常,這與其他語(yǔ)言不同,異常使用 try…cache 觸發(fā)。
手動(dòng)拋出異常的方法:
throw new 錯(cuò)誤類(lèi)型($message, $code);
注意:可以設(shè)置多個(gè)catch,參數(shù)Exception 或者繼承Exception的類(lèi)
Exception 類(lèi)的方法:
方法 | 說(shuō)明 | 重寫(xiě) |
---|---|---|
getFile | 產(chǎn)生異常錯(cuò)誤的文件 | NO,final |
getCode | 錯(cuò)誤碼 | NO,final |
getLine | 錯(cuò)誤行號(hào) | NO,final |
getMessage | 錯(cuò)誤消息 | NO,final |
__toString | 對(duì)象轉(zhuǎn)字符串后輸出內(nèi)容 | YES |
finally: 需要放在 catch
后,finally
無(wú)論是否拋出異常都會(huì)執(zhí)行。
登錄異常處理的例子:
class LoginException extends Exception { // 構(gòu)造函數(shù) public function __construct($message, $code = 0, Throwable $previous = null) { parent::__construct($message, $code, $previous); } // 將異常對(duì)象轉(zhuǎn)為字符串 public function __toString() { return __CLASS__ . ": [{$this->code}]: {$this->message}\n"; } // 自定義異常處理函數(shù) public function logError() { // 記錄登錄錯(cuò)誤日志 error_log("登錄異常: " . $this->getMessage()); } } function loginUser($username, $password) { // 模擬登錄邏輯 if ($username !== "admin" || $password !== "password") { throw new LoginException("無(wú)效的用戶名或密碼。"); } // 登錄成功 return true; } try { $loggedIn = loginUser("guest", "pass123"); // 嘗試登錄 if ($loggedIn) { echo "登錄成功!"; } } catch (LoginException $e) { // 捕獲登錄異常 echo "登錄失?。? . $e->getMessage() . "\n"; $e->logError(); // 調(diào)用自定義異常處理函數(shù),記錄登錄錯(cuò)誤日志 } catch (Exception $e) { // 捕獲其他異常 echo "發(fā)生了一個(gè)錯(cuò)誤:" . $e->getMessage() . "\n"; }
操作數(shù)據(jù)庫(kù)多表聯(lián)動(dòng)時(shí)捕獲異常回滾數(shù)據(jù):
try { // 連接數(shù)據(jù)庫(kù) $db = new PDO("mysql:host=localhost;dbname=mydatabase", "username", "password"); // 開(kāi)始事務(wù) $db->beginTransaction(); // 執(zhí)行一系列數(shù)據(jù)庫(kù)操作 $db->exec("INSERT INTO users (name, email) VALUES ('John Doe', 'john@example.com')"); $db->exec("INSERT INTO orders (user_id, product_id, quantity) VALUES (1, 1, 2)"); // 提交事務(wù) $db->commit(); } catch (PDOException $e) { // 捕獲 PDO 異常 // 回滾事務(wù) $db->rollBack(); // 記錄異常日志 error_log("數(shù)據(jù)庫(kù)操作異常: " . $e->getMessage()); // 輸出錯(cuò)誤信息 echo "發(fā)生了一個(gè)數(shù)據(jù)庫(kù)操作異常:" . $e->getMessage() . "\n"; }
總結(jié):
異常處理應(yīng)該是有針對(duì)性的:捕獲和處理那些你知道如何處理的異常類(lèi)型,而將未知或不可處理的異常傳遞給更上層的代碼進(jìn)行處理。
使用合適的異常類(lèi)型:PHP 中有許多內(nèi)置的異常類(lèi)型,如
Exception
、InvalidArgumentException
、RuntimeException
等。選擇合適的異常類(lèi)型可以更好地組織和識(shí)別代碼中的異常情況。按照異常類(lèi)型的順序進(jìn)行捕獲:在
catch
塊中,應(yīng)該按照特定異常類(lèi)型到更一般異常類(lèi)型的順序進(jìn)行排序。這樣可以確保異常被正確捕獲,并且避免特定的異常被更一般的異常類(lèi)型捕獲導(dǎo)致無(wú)法執(zhí)行特定的異常處理邏輯。使用多個(gè)
catch
塊進(jìn)行處理:根據(jù)不同的異常類(lèi)型,使用多個(gè)catch
塊來(lái)處理不同類(lèi)型的異常。這樣可以更細(xì)粒度地處理異常,并根據(jù)需要執(zhí)行特定的邏輯。考慮添加日志或錯(cuò)誤記錄:在異常處理過(guò)程中,可以將異常信息記錄到日志文件或錯(cuò)誤日志中,以便后續(xù)的故障排查和分析。
避免過(guò)度捕獲異常:不要過(guò)度捕獲異常,這可能導(dǎo)致隱藏潛在的問(wèn)題或掩蓋錯(cuò)誤。只捕獲你知道如何處理的異常,并將其他未處理的異常傳遞給更上層的代碼進(jìn)行處理。
使用
finally
塊(可選):finally
塊中的代碼無(wú)論是否發(fā)生異常都會(huì)執(zhí)行。它通常用于釋放資源或執(zhí)行清理操作,以確保在異常發(fā)生時(shí)資源得到正確處理。拋出自定義異常:通過(guò)定義自定義異常類(lèi),可以更好地組織和標(biāo)識(shí)特定類(lèi)型的異常。這樣可以使代碼更加清晰,并提供更豐富的異常信息。
PHP常見(jiàn)問(wèn)題合集開(kāi)發(fā)中的錯(cuò)誤處理與日志記錄
本文將總結(jié)常見(jiàn)的PHP開(kāi)發(fā)問(wèn)題,并提供錯(cuò)誤處理和日志記錄的解決方案。
一、PHP開(kāi)發(fā)中的常見(jiàn)問(wèn)題
語(yǔ)法錯(cuò)誤:在PHP開(kāi)發(fā)中,常常會(huì)出現(xiàn)拼寫(xiě)錯(cuò)誤、缺少分號(hào)等語(yǔ)法錯(cuò)誤。這些錯(cuò)誤會(huì)導(dǎo)致程序無(wú)法正常執(zhí)行。為了解決這些問(wèn)題,我們應(yīng)該養(yǎng)成良好的編碼習(xí)慣,并且在開(kāi)發(fā)過(guò)程中使用代碼編輯器的自動(dòng)補(bǔ)全和語(yǔ)法檢查功能。
變量未定義或未初始化:在PHP中,如果使用未定義或未初始化的變量,會(huì)導(dǎo)致警告或致命錯(cuò)誤。為了避免這種問(wèn)題,應(yīng)該始終在使用變量之前對(duì)其進(jìn)行初始化或定義。
函數(shù)不存在或參數(shù)錯(cuò)誤:在調(diào)用函數(shù)時(shí),如果函數(shù)不存在或者傳遞了錯(cuò)誤的參數(shù),會(huì)導(dǎo)致程序無(wú)法正常執(zhí)行。為了解決這些問(wèn)題,應(yīng)該對(duì)函數(shù)的存在進(jìn)行檢查,并確保傳遞正確的參數(shù)。
數(shù)據(jù)庫(kù)連接問(wèn)題:在使用PHP開(kāi)發(fā)Web應(yīng)用程序時(shí),經(jīng)常會(huì)涉及到與數(shù)據(jù)庫(kù)的連接。如果連接失敗或出現(xiàn)其他問(wèn)題,會(huì)導(dǎo)致無(wú)法正常訪問(wèn)或操作數(shù)據(jù)庫(kù)。為了解決這些問(wèn)題,應(yīng)該確保數(shù)據(jù)庫(kù)的配置正確,并使用try-catch語(yǔ)句捕獲異常。
文件訪問(wèn)權(quán)限問(wèn)題:在PHP開(kāi)發(fā)中,經(jīng)常需要讀取或?qū)懭胛募H绻募脑L問(wèn)權(quán)限不正確,會(huì)導(dǎo)致無(wú)法讀取或?qū)懭胛募?。為了解決這些問(wèn)題,應(yīng)該確保文件有足夠的訪問(wèn)權(quán)限,并在操作文件時(shí)進(jìn)行錯(cuò)誤處理。
二、錯(cuò)誤處理的方法
錯(cuò)誤報(bào)告設(shè)置:在開(kāi)發(fā)環(huán)境中,應(yīng)該將PHP的錯(cuò)誤報(bào)告級(jí)別設(shè)置為E_ALL,以便能夠捕獲所有錯(cuò)誤。在生產(chǎn)環(huán)境中,應(yīng)該將錯(cuò)誤報(bào)告級(jí)別設(shè)置為E_ALL & ~E_NOTICE,以避免顯示一些無(wú)關(guān)緊要的通知。
異常處理:在PHP中,可以使用try-catch語(yǔ)句來(lái)捕獲異常并進(jìn)行處理。在拋出異常時(shí),應(yīng)該提供有意義的錯(cuò)誤消息,并在catch塊中進(jìn)行相應(yīng)的處理,例如記錄日志或顯示錯(cuò)誤信息給用戶。
自定義錯(cuò)誤處理函數(shù):可以使用set_error_handler()函數(shù)來(lái)設(shè)置自定義的錯(cuò)誤處理函數(shù)。在該函數(shù)中,可以定義錯(cuò)誤消息的輸出方式,例如將錯(cuò)誤消息寫(xiě)入日志文件或發(fā)送郵件通知。
三、日志記錄的方法
使用日志庫(kù):PHP中有許多開(kāi)源的日志庫(kù),例如Monolog和Log4php。通過(guò)使用這些庫(kù),可以方便地將錯(cuò)誤信息寫(xiě)入文件、數(shù)據(jù)庫(kù)或其他目標(biāo)。
日志級(jí)別設(shè)置:在日志記錄中,可以設(shè)置不同的日志級(jí)別,如DEBUG、INFO、WARNING、ERROR和CRITICAL等。根據(jù)不同的情況,可以選擇記錄不同級(jí)別的日志信息。
日志分割和歸檔:為了避免日志文件過(guò)大,可以使用日志分割和歸檔功能。通過(guò)使用日志庫(kù)提供的工具,可以按照時(shí)間或文件大小等條件,將日志文件分割成多個(gè)文件,并進(jìn)行歸檔。
總結(jié)起來(lái),開(kāi)發(fā)中會(huì)遇到各種PHP問(wèn)題,如語(yǔ)法錯(cuò)誤、變量未定義、函數(shù)不存在、數(shù)據(jù)庫(kù)連接問(wèn)題和文件訪問(wèn)權(quán)限問(wèn)題等。為了解決這些問(wèn)題,可以使用錯(cuò)誤處理和日志記錄的方法。錯(cuò)誤處理包括設(shè)置錯(cuò)誤報(bào)告、使用異常處理和自定義錯(cuò)誤處理函數(shù)。日志記錄包括使用日志庫(kù)、設(shè)置日志級(jí)別和實(shí)現(xiàn)日志分割和歸檔等。通過(guò)合理使用這些方法,可以提高代碼的質(zhì)量和可維護(hù)性,從而改善開(kāi)發(fā)效率和用戶體驗(yàn)。
相關(guān)文章
PHP實(shí)現(xiàn)word轉(zhuǎn)pdf的兩種方式(有用!)
最近遇到一個(gè)需求,將word文件轉(zhuǎn)換成pdf,輾轉(zhuǎn)了一上午終于搞定了,下面這篇文章主要給大家介紹了關(guān)于PHP實(shí)現(xiàn)word轉(zhuǎn)pdf的兩種方式,需要的朋友可以參考下2022-09-09無(wú)JS,完全php面向過(guò)程數(shù)據(jù)分頁(yè)實(shí)現(xiàn)代碼
無(wú)JS,完全php面向過(guò)程數(shù)據(jù)分頁(yè)實(shí)現(xiàn)代碼,需要的朋友可以參考下2012-08-08PHP var關(guān)鍵字相關(guān)原理及使用實(shí)例解析
這篇文章主要介紹了PHP var關(guān)鍵字相關(guān)原理及使用實(shí)例解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-07-07PHP實(shí)現(xiàn)獲取兩個(gè)以逗號(hào)分割的字符串的并集
這篇文章主要為大家詳細(xì)介紹了PHP如何實(shí)現(xiàn)獲取兩個(gè)以逗號(hào)分割的字符串的并集,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2023-04-04php smarty 二級(jí)分類(lèi)代碼和模版循環(huán)例子
分享下最近寫(xiě)的smarty模版引擎輸出二級(jí)分類(lèi)代碼,主要是靠二維數(shù)組進(jìn)行控制輸出。2011-06-06PHP中soap用法示例【SoapServer服務(wù)端與SoapClient客戶端編寫(xiě)】
這篇文章主要介紹了PHP中soap用法,結(jié)合實(shí)例形式分析了SoapServer服務(wù)端與SoapClient客戶端相關(guān)實(shí)現(xiàn)技巧與操作注意事項(xiàng),需要的朋友可以參考下2018-12-12php實(shí)現(xiàn)Mongodb自定義方式生成自增ID的方法
這篇文章主要介紹了php實(shí)現(xiàn)Mongodb自定義方式生成自增ID的方法,實(shí)例分析了Mongodb自增字段的實(shí)現(xiàn)技巧與對(duì)應(yīng)php操作方法,需要的朋友可以參考下2015-03-03php+Ajax處理xml與json格式數(shù)據(jù)的方法示例
這篇文章主要介紹了php+Ajax處理xml與json格式數(shù)據(jù)的方法,結(jié)合實(shí)例形式分析了ajax處理xml格式數(shù)據(jù)及json格式數(shù)據(jù)相關(guān)操作技巧與注意事項(xiàng),需要的朋友可以參考下2019-03-03