自定義錯誤報告的處理方式,可以完全繞過標准的PHP錯誤處理函數,這樣就可以按照自己定義的格式打印錯誤報告,或改變錯誤報告打印的位置(標准PHP的錯誤報告是哪里發生錯誤就在發生位置處顯示)。以下幾種情況可以考慮自定義錯誤處理。
★可以記下錯誤的信息,及時發現一些生產環境出現的問題。
★可以用來屏蔽錯誤。出現錯誤會把一些信息暴漏給用戶,極有可能成為黑客攻擊你網站的工具。
★可以做相應的處理,將所有錯誤報告放到腳本最后輸出,或出錯時可以顯示跳轉到預先定義好的出錯頁面,提供更好的用戶體驗,如果必要,還可以在自定義錯誤處理程序中,根據情況去終止腳本運行。
★可以作為調試工具,一些時候必須在運行環境時調試一些東西,但又不想影響正在使用的用戶。
通常使用set_error_handler()函數去設置用戶自定義的錯誤處理函數,該函數用於創建運行時期間的用戶自己的錯誤處理方法,返回舊的錯誤處理程序,若失敗,則返回null。該函數有兩個參數,其中第一個參數是必選的,需要一個回調函數,規定發生錯誤時運行的函數。這個回調函數一定要聲明4個參數,否則無效,按順序分別為“是否存在錯誤”、“錯誤信息”、“錯誤文件”和“錯誤行號”。set_error_handler()函數的第二個參數則為可選的,規定現在哪個錯誤報告級別會顯示用戶自定義的錯誤。默認是“E_ALL”。自定義錯誤處理的示例如下所示:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
|
<?php
error_reporting
(0);
//屏蔽程序中的錯誤
//定義Error_Handler函數,作為set_error_handler()函數的第一個參數“回調”
function
error_handler(
$error_level
,
$error_message
,
$file
,
$line
){
$EXIT
=FALSE;
switch
(
$error_level
){
//提醒級別
case
E_NOTICE:
case
E_USER_NOTICE:
$error_type
=
'Notice'
;
break
;
//警告級別
case
E_WARNING:
case
E_USER_WARNING:
$error_type
=
'warning'
;
break
;
//錯誤級別
case
E_ERROR:
case
E_USER_ERROR:
$error_type
=
'Fatal Error'
;
$EXIT
= TRUE;
break
;
//其他未知錯誤
default
:
$error_type
=
'Unknown'
;
$EXIT
= TRUE;
break
;
}
//直接打印錯誤信息,也可以寫文件,寫數據庫,反正錯誤信息都在這,任你發落
printf(
"<font color='#FF0000'><b>%s</b></font>:%s in<b>%s</b> on line <b>%d</b><br>\n"
,
$error_type
,
$error_message
,
$file
,
$line
);
//如果錯誤影響到程序的正常執行,跳轉到友好的錯誤提示頁面
if
(TURE==
$EXIT
){
echo
'<script>location = "err.html";</scrpit>'
;
}
}
//這個才是關鍵點,把錯誤的處理交給error_handle()
set_error_handler(
'error_handler'
);
//使用未定義的變量要報notice的
echo
$novar
;
//除以0要報警告的
echo
3/0;
//自定義一個錯誤
trigger_error(
'Trigger a fatal error'
,E_USER_ERROR);
?>
|
本例所有打印的錯誤報告都是按自己定義的格式輸出的,不過有一點,系統直接報Fatal Error的這里捕獲不到,因為系統不可能把這么重大的錯誤教給你處理。遇到這種錯誤是必須要解決的,所以系統會直接終止程序運行。使用set_error_handler()函數可以很好地解決安全和調試方便的矛盾,而且你還可以花點心思,使錯誤提示更加美觀以配合網站的風格。不過要注意兩點。
①E_ERROR、E_PARSE、E_CORE_ERROR、E_CORE_WARNING、E_COMPILE_ERROR、E_COMPILE_WARNING是不會被這個句柄處理的,也就是會用最原始的方式顯示出來。不過出現這些錯誤都是編譯或PHP內核出錯,在通常情況下不會發生。
②使用set_error_handler()后,error_reporting()將會失效。也就是所有的錯誤(除上述的錯誤)都會教給自己定義的函數處理。