異常與錯誤
異常是指程序運(yùn)行中不符合預(yù)期情況以及與正常流程不同的狀況雾家。錯誤則屬于自身問題,是一種非法語法或者環(huán)境問題導(dǎo)致的芯咧、讓編譯器無法通過檢查設(shè)置無法運(yùn)行的情況。
由于php最開始是沒有異常處理邪铲,后來為了進(jìn)軍企業(yè)級開發(fā)无拗,模仿java等語言,推出了異常揽惹。導(dǎo)致php中遇到任何自身錯誤都會觸發(fā)一個錯誤四康,而不是拋出一個異常(某些情況下,會同時拋出錯誤和異常)疯溺。PHP一旦遇到非正常代碼,大多數(shù)情況下囱嫩,都是直接拋出錯誤,而不是異常今妄。
php只有在你throw 一個異常后损俭,才能用try...catch來捕獲異常(一般情況下如此潘酗,也有部分異常可以自動捕獲)琐脏。
在php中通常會在以下場景中使用異常:
對程序的悲觀預(yù)測:如果認(rèn)為自己的代碼無法一一處理各種可預(yù)見的情況缸兔、不可預(yù)見的情況。
程序的需要和對業(yè)務(wù)的關(guān)注 : 如果對數(shù)據(jù)的一致性要求很高時昂拂,可以用try...catch把異常造成的邏輯中斷破壞將到最小抛猖,并且經(jīng)過補(bǔ)救處理后,不影響業(yè)務(wù)邏輯的完整性蛉抓。
語言級別的健壯性要求 : 通過精確控制運(yùn)行時的流程,在程序中斷時朝墩,有預(yù)見的用try...catch縮小可能出錯的范圍伟姐,及時捕獲異常并做出相應(yīng)的補(bǔ)救。
怎樣看待php的異常
歷史原因?qū)е聀hp的異常處理是不足的鹿霸,絕大多數(shù)情況下恐似,無法自動拋出異常,必須使用if...else先進(jìn)行判斷葛闷,再手動拋出異常。
手動拋出異常的意義不是很大淑趾,因?yàn)檫@意味著在代碼里已經(jīng)充分的預(yù)期到錯誤的出現(xiàn)近范。同時這種方式還會讓你在復(fù)雜的邏輯判斷和處理中暈頭轉(zhuǎn)向延蟹。導(dǎo)致失去異常真正的優(yōu)點(diǎn)阱飘。
那么有更好的異常拋出方法嗎沥匈?有,那就是結(jié)合使用錯誤
php中的錯誤
錯誤就是會使腳本運(yùn)行不正常的情況缰儿。
在php中主要的錯誤等級如下:
deprecated: 最低級別的錯誤,表示"不推薦, 不建議"乖阵。例如在php 5中使用了ereg系列的正則函數(shù)就會出現(xiàn)。這類錯誤一般由于使用了不推薦的义起、過時的函數(shù)或語法造成。不影響程序正常運(yùn)行师崎,但建議修正默终。
notice: 一般指語法中存在不恰當(dāng)?shù)牡胤健H缡褂米兞康俏炊x就會報此錯誤犁罩。不影響程序正常流程。
warning: 較高級別的錯誤含滴,在語法中出現(xiàn)很不恰當(dāng)?shù)那闆r才會出現(xiàn)此錯誤丐巫,比如函數(shù)參數(shù)不匹配勺美。會導(dǎo)致得不到預(yù)期的結(jié)果赡茸,需要修改代碼。
fetal error: 致命錯誤占卧,直接導(dǎo)致程序終止運(yùn)行联喘。這類錯誤必須修改。
prase error: 語法解析錯誤豁遭,上面幾種都屬于運(yùn)行時錯誤,此錯誤在運(yùn)行前就會拋出域滥。
在php中蜈抓,總共有16錯誤級別昂儒,但是主要的就是上面幾種。
error.php
$data = '2012-12-20';
if (ereg("([0-9]{4})-([0-9]{1,2})-([0-9]{1,2})", $data, $regs)) {
echo "$reg[3].$regs[2].$regs[1]";
} else {
echo "Invalid data format: $data";
}
$a = array('o' => 2, 4, 6, 8);
echo $a[o];
$result = array_sum($a, 3);
echo func();
echo '致命錯誤后腊嗡,還會執(zhí)行嗎拾酝?';
//echo '最高級別錯誤', $55;
上面代碼執(zhí)行后,會有四個錯誤級別蒿囤,如果你無法完全看到的話材诽,你需要去修改你的ini配置文件中錯誤顯示級別為 E_ALL
自定義錯誤處理程序
可以使用 set_error_handler() 函數(shù)來托管錯誤處理程序,可自行定制錯誤的處理流程脸侥。
如果要取消托管的話,可以在同一個頁面中使用restore_error_handler()來取消托管睁枕。
如果想要自己拋出一個錯誤的話沸手,可以使用trigger_error()函數(shù)罐氨。
<?php
//自定義錯誤處理程序
function customError($errno, $errstr, $errfile, $errline)
{
echo "<b>錯誤代碼:</b>[{$errno}] {$errstr}", PHP_EOL;
echo "錯誤所在代碼行:{$errline} 文件{$errfile}", PHP_EOL;
echo "PHP版本", PHP_VERSION, "(", PHP_OS, ")", PHP_EOL;
}
set_error_handler("customError", E_ALL | E_STRICT);
$a = array('o' => 2, 4, 6, 8);
echo $a[o];
執(zhí)行上面的代碼滩援,可以看到錯誤信息是由我們自定義的處理程序輸出的,完全繞開了系統(tǒng)的處理程序租悄。
如果錯誤發(fā)生在自定義處理程序前恩袱,則不會調(diào)用我們自定義的錯誤處理程序,所以應(yīng)當(dāng)先定義錯誤處理程序畔塔。
當(dāng)然不是所有的錯誤級別都可以用set_error_handler來托管,如E_ERROR把敢、E_PARSE谅辣、E_CODE_WARNING、E_COMPILE_ERROR桑阶、E_COMPILE_WARNING以及E_STRICT中的部分蚣录。這些錯誤信息會以原始的方式來顯示或者不現(xiàn)實(shí)。
PHP把許多異澄樱看作是錯誤,所以這些"異常"同樣可以使用set_error_handler來接管:
function customError($errno, $errstr, $errfile, $errline)
{
//自定義錯誤處理是换可,手動拋出異常
throw new Exception($errstr);
}
set_error_handler('customError', E_ALL | E_STRICT);
try {
$a = 5/0;
} catch (Exception $e) {
echo '錯誤信息:', $e->getMessage();
}
當(dāng)然這種處理方式也有自己的優(yōu)缺點(diǎn):
缺點(diǎn): 必須依靠程序員自己來掌控對異常的處理厦幅,對于異常的高發(fā)區(qū)、敏感區(qū)译荞,如果程序員處理不好,就會導(dǎo)致業(yè)務(wù)數(shù)據(jù)不一致的問題吞歼。
優(yōu)點(diǎn): 可以獲得程序運(yùn)行時的上下文信息篙骡,以進(jìn)行針對性的補(bǔ)救。
fetal error這樣的錯誤無法捕獲糯俗,也無法在發(fā)生后恢復(fù)流程處理,但是可以使用register_shutdown_function()函數(shù)在程序終止或die時觸發(fā)一個函數(shù)杖玲,給程序帶來一個短暫的回光返照淘正。在php4時,不支持析構(gòu)函數(shù)鸿吆,也常用于模擬實(shí)現(xiàn)析構(gòu)函數(shù)伞剑。
class Shutdown
{
public function stop()
{
if (error_get_laster()) {
print_r(error_get_laster());
}
die('Stop.');
}
}
register_shutdown_function(array(new Shutdown(), 'stop'));
$a = new a(); //致命錯誤,導(dǎo)致失敗
echo '必須終止';
Parse error級別的錯誤黎泣,除了修改ini文件抒倚,將錯誤信息寫到日志中托呕,什么也做不了频敛。
小結(jié)
php中錯誤和異常是兩個不同的概念,這種設(shè)計根本上導(dǎo)致了php的異常和錯誤與其它語言相異着降。java中拗军,異常時錯誤唯一的報告方式蓄喇。說到底交掏,兩者的區(qū)別就是對異常的認(rèn)識不同產(chǎn)生的。php異常絕大部分是通過某種方式手動拋出钱骂,才能捕獲到挪鹏。是一種半自動化的異常處理機(jī)制。
無論是錯誤還是異常狰住,都可以使用handler接管系統(tǒng)已有的處理機(jī)制。