Web應(yīng)用中對(duì)于異常的處理方式與其他形式的應(yīng)用并沒(méi)有太大的不同――通過(guò)try/catch
語(yǔ)句針對(duì)不同的異常進(jìn)行相應(yīng)處理。
但是在具體實(shí)現(xiàn)中,由于異常層次、種類繁雜,我們往往很難在Servlet、JSP層妥善的處
理好所有異常情況,代碼中大量的try/catch代碼顯得尤為凌亂。
我們通常面對(duì)下面兩個(gè)主要問(wèn)題:
1. 對(duì)異常實(shí)現(xiàn)集中式處理
典型情況:對(duì)數(shù)據(jù)庫(kù)異常記錄錯(cuò)誤日志。一般處理方法無(wú)外兩種,一是在各處數(shù)據(jù)庫(kù)
訪問(wèn)代碼的異常處理中,加上日志記錄語(yǔ)句。二是將在數(shù)據(jù)訪問(wèn)代碼中將異常向上拋
出,并在上層結(jié)構(gòu)中進(jìn)行集中的日志記錄處理。
第一種處理方法失之繁瑣、并且導(dǎo)致系統(tǒng)難以維護(hù),假設(shè)后繼需求為“對(duì)于數(shù)據(jù)庫(kù)異
常,需記錄日志,并發(fā)送通知消息告知系統(tǒng)管理員”。我們不得不對(duì)分散在系統(tǒng)中的各
處代碼進(jìn)行整改,工作量龐大。
第二種處理方法實(shí)現(xiàn)了統(tǒng)一的異常處理,但如果缺乏設(shè)計(jì),往往使得上層異常處理過(guò)
于復(fù)雜。
這里,我們需要的是一個(gè)設(shè)計(jì)清晰、成熟可靠的集中式異常處理方案。
2. 對(duì)未捕獲異常的處理
對(duì)于Unchecked Exception而言,由于代碼不強(qiáng)制捕獲,往往被程序員所忽略,如果
運(yùn)行期產(chǎn)生了Unchecked Exception,而代碼中又沒(méi)有進(jìn)行相應(yīng)的捕獲和處理,則我
們可能不得不面對(duì)尷尬的500服務(wù)器內(nèi)部錯(cuò)誤提示頁(yè)面。
這里,我們需要一個(gè)全面而有效的異常處理機(jī)制。
上面這兩個(gè)問(wèn)題,從技術(shù)角度上而言并算不上什么大的難點(diǎn)。套用一些短平快的設(shè)計(jì)模式,
我們也能進(jìn)行處理并獲得不錯(cuò)的效果。同時(shí),目前大多數(shù)服務(wù)器也都支持在Web.xml中通過(guò)
<error-page>(Websphere/Weblogic)或者<error-code>(Tomcat)節(jié)點(diǎn)配置特定異常情
況的顯示頁(yè)面。
Spring MVC中提供了一個(gè)通用的異常處理機(jī)制,它提供了一個(gè)成熟的,簡(jiǎn)潔清晰的異常處
理方案。如果基于Spring MVC開(kāi)發(fā)Web應(yīng)用,那么利用這套現(xiàn)成的機(jī)制進(jìn)行異常處理也更加自
然和有效。
Spring MVC中的異常處理:
以前面的注冊(cè)系統(tǒng)為例,首先,在Dispatcher配置文件Config.xml中增加id為
“exceptionResolver”的bean定義:
<bean id="exceptionResolver"
class="org.springframework.web.servlet.handler.SimpleMappingEx
ceptionResolver">
<property name="defaultErrorView">
<value>failure</value>
</property>
<property name="exceptionMappings">
<props>
<prop key="java.sql.SQLException">showDBError</prop>
<prop key="java.lang.RuntimeException">showError</prop>
</props>
</property>
</bean>
通過(guò)SimpleMappingExceptionResolver我們可以將不同的異常映射到不同的jsp頁(yè)
面(通過(guò)exceptionMappings屬性的配置),同時(shí)我們也可以為所有的異常指定一個(gè)默認(rèn)的異
常提示頁(yè)面(通過(guò)defaultErrorView屬性的配置),如果所拋出的異常在exceptionMappings
中沒(méi)有對(duì)應(yīng)的映射,則Spring將用此默認(rèn)配置顯示異常信息(注意這里配置的異常顯示界面均
僅包括主文件名,至于文件路徑和后綴已經(jīng)在viewResolver中指定)。
一個(gè)典型的異常顯示頁(yè)面如下:
<html>
<head><title>Exception!</title></head>
<body>
<% Exception ex = (Exception)request.getAttribute("exception"); %>
<H2>Exception: <%= ex.getMessage();%></H2>
<P/>
<% ex.printStackTrace(new java.io.PrintWriter(out)); %>
</body>
</html>
exception 實(shí)在SimpleMappingExceptionResolver 被存放到request中的,具體可以查看源代碼。
如果SimpleMappingExceptionResolver無(wú)法滿足異常處理的需要,我們可以針對(duì)
HandlerExceptionResolver接口實(shí)現(xiàn)自己異常處理類,這同樣非常簡(jiǎn)單(只需要實(shí)現(xiàn)一個(gè)
resolveException方法)。
如果有ViewResolver,則制定的jsp頁(yè)面必須在那個(gè)頁(yè)面下,到時(shí)候如果找不到頁(yè)面,可以根據(jù)錯(cuò)誤提示再調(diào)整頁(yè)面路徑