PHP批斗大會(huì)之缺失的異常詳解
故事的開(kāi)始
這幾天觀察錯(cuò)誤日志發(fā)現(xiàn)有一個(gè)數(shù)據(jù)反序列化的notice錯(cuò)誤,實(shí)際情況我是從緩存中讀取數(shù)據(jù)然后反序列化,因?yàn)榉葱蛄谢?,所以?shí)際每次都是去數(shù)據(jù)庫(kù)取的值。背后性能影響還是挺大的。
缺失的異常
剛開(kāi)始寫(xiě)代碼的時(shí)候一直不明白為什么要用異常,感覺(jué)if else就能搞定了,為什么還要多此一舉,現(xiàn)在反而覺(jué)得 php 的異常太少。
對(duì)比兩種序列化場(chǎng)景,一個(gè)是json,另一個(gè)是serialize。
json
在json encode/decode的時(shí)候,如果出現(xiàn)異常,可以通過(guò)json_last_error()來(lái)獲取。
https://www.php.net/manual/en...
這樣的設(shè)計(jì)只能說(shuō)勉強(qiáng)夠用,不太符合面向?qū)ο蟮奶茁贰?/p>
serialize/unserialize
在使用自帶的序列化和反序列化的時(shí)候,相比json的處理,則更加簡(jiǎn)單粗暴,沒(méi)有函數(shù)能拿到最后的錯(cuò)誤,只會(huì)通過(guò)自定義的error handler來(lái)接管,然后自己去做出一些相應(yīng)的處理。
為什么要捕獲異常
比如我的代碼比較亂,有的 key 是 json 序列化,有的 key 是 serialize。我們可以將 key 分類(lèi)。不能確保其他人配置的對(duì)應(yīng)關(guān)系是對(duì)的,或者有的人忘記了,所以我需要用捕獲異常的方式來(lái)兜底,這樣我們的代碼更加健壯一些。當(dāng)unserialize失敗之后,我們可以嘗試去json_decode,而不是立即返回一個(gè)false,從而把請(qǐng)求傳遞到數(shù)據(jù)庫(kù)。
代碼演示
error_reporting(E_ALL); $a = ["a" => 1]; class UnSerializeException extends ErrorException { } set_error_handler(function ($severity, $message, $file, $line) { $info = explode(":", $message); if ($severity == E_NOTICE) { if ($info[0] == "unserialize()") { throw new UnSerializeException($message); } return true; } else { throw new ErrorException($message, 0, $severity, $file, $line);; } }); try { $b = unserialize(json_encode($a)); } catch (ErrorException $exception) { var_dump(get_class($exception), $exception->getMessage(), $exception->getTraceAsString()); // 捕獲到了 } finally { restore_error_handler(); } try { $b = unserialize(json_encode($a)); } catch (ErrorException $exception) { var_dump(get_class($exception), $exception->getMessage(), $exception->getTraceAsString()); // 無(wú)法捕獲 }
輸出結(jié)果
string(20) "UnSerializeException"
string(43) "unserialize(): Error at offset 0 of 7 bytes"
string(181) "#0 [internal function]: {closure}(8, 'unserialize(): ...', '/Users/mengkang...', 34, Array)
#1 /Users/mengkang/PhpstormProjects/xxx/test.php(34): unserialize('{"a":1}')
#2 {main}"Notice: unserialize(): Error at offset 0 of 7 bytes in /Users/mengkang/PhpstormProjects/xxx/test.php on line 42
后記
所以 php 代碼的異常設(shè)計(jì)還是任重而道遠(yuǎn)的,而這些已經(jīng)設(shè)定的“舊的規(guī)范”要推翻,需要“勇氣”,畢竟會(huì)影響所有的使用者。
好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
解決安裝WampServer時(shí)提示缺少msvcr110.dll文件的問(wèn)題
下面小編就為大家?guī)?lái)一篇解決安裝WampServer時(shí)提示缺少msvcr110.dll文件的問(wèn)題。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-07-07PHP使用Session實(shí)現(xiàn)上傳進(jìn)度功能詳解
這篇文章主要介紹了PHP使用Session實(shí)現(xiàn)上傳進(jìn)度功能,結(jié)合實(shí)例形式詳細(xì)分析了session上傳進(jìn)度顯示相關(guān)原理、實(shí)現(xiàn)方法及操作注意事項(xiàng),需要的朋友可以參考下2019-08-08Mac環(huán)境下php操作mysql數(shù)據(jù)庫(kù)的方法分享
今天在mac上搭建好了php的環(huán)境,我們就把php操作mysql數(shù)據(jù)庫(kù)的方法分享給大家,有需要的小伙伴參考下。2015-05-05利用PHP如何統(tǒng)計(jì)Nginx日志的User Agent數(shù)據(jù)
這篇文章主要給大家介紹了關(guān)于如何利用PHP統(tǒng)計(jì)Nginx日志的User Agent數(shù)據(jù)的相關(guān)資料,文中通過(guò)示例代碼以及圖文介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03php中刪除字符串中最先出現(xiàn)某個(gè)字符的實(shí)現(xiàn)代碼
刪除字符串中最先出現(xiàn)某個(gè)字,就是通過(guò)explode的靈活用法,需要的朋友可以參考下2013-02-02php session_start()出錯(cuò)原因分析及解決方法
本文是對(duì)php中session_start()的出錯(cuò)原因及解決方法進(jìn)行了詳細(xì)的介紹,需要的朋友可以過(guò)來(lái)參考下,希望對(duì)大家有所幫助2013-10-10PHP遞歸實(shí)現(xiàn)層級(jí)樹(shù)狀展開(kāi)
這篇文章主要為大家詳細(xì)介紹了PHP遞歸實(shí)現(xiàn)層級(jí)樹(shù)狀展開(kāi)的相關(guān)資料,需要的朋友可以參考下2016-04-04