Struts異常處理(原創(chuàng)) 異 常是當(dāng)JVM(JAVA虛擬機(jī))在執(zhí)行應(yīng)用程序的某個(gè)方法的時(shí)候遇到的非正常現(xiàn)象,JVM就會(huì)生成一個(gè)異常對(duì)象,拋給客戶以便客戶進(jìn)行異常處理. Struts框架的異常處理機(jī)制建立在java異常處理的基礎(chǔ)之上.在研究Struts異常處理之前,先回顧一下java異常處理原理,理解java虛擬 機(jī)JVM的異常處理過程有助于應(yīng)用設(shè)計(jì)正確的異常處理方法.處理異常需要JVM付出不小的開銷,所以用于必須慎重使用. JAVA 異常是在java程序運(yùn)行的時(shí)候遇到非正常的情況而創(chuàng)建的對(duì)象,它封裝了異常信息,java異常的根類為java.lang.Throwable,整個(gè)類 有兩個(gè)直接子類java.lang.Error和java.lang.Exception.Error是程序本身無法恢復(fù)的嚴(yán)重錯(cuò)誤.Exception 則表示可以被程序捕獲并處理的異常錯(cuò)誤.JVM用方法調(diào)用棧來跟蹤每個(gè)線程中一系列的方法調(diào)用過程,該棧保存了每個(gè)調(diào)用方法的本地信息.對(duì)于獨(dú)立的 JAVA程序,可以一直到該程序的main方法.當(dāng)一個(gè)新方法被調(diào)用的時(shí)候,JVM把描述該方法的棧結(jié)構(gòu)置入棧頂,位于棧頂?shù)姆椒檎_執(zhí)行的方法.當(dāng)一 個(gè)JAVA方法正常執(zhí)行完畢,JVM回從調(diào)用棧中彈處該方法的棧結(jié)構(gòu),然后繼續(xù)處理前一個(gè)方法.如果java方法在執(zhí)行代碼的過程中拋出異常,JVM必須 找到能捕獲異常的catch塊代碼.它首先查看當(dāng)前方法是否賊這樣的catch代碼塊,如果存在就執(zhí)行該catch代碼塊,否則JVM回調(diào)用棧中彈處該方 法的棧結(jié)構(gòu),繼續(xù)到前一個(gè)方法中查找合適的catch代碼塊.最后如果JVM向上追到了main()方法,也就是一直把異常拋給了main()方法,仍然 沒有找到該異常處理的代碼塊,該線程就會(huì)異常終止,如果該線程是主線程,應(yīng)用程序也隨之終止,此時(shí)JVM將把異常直接拋給用戶,在用戶終端上會(huì)看到原始的 異常信息.回顧完了JAVA的異常處理機(jī)制,就要開始研究Struts的異常處理了. Struts 框架在視圖層和控制層提供了對(duì)異常處理的支持.Struts的控制器負(fù)責(zé)捕獲各種異常,包括控制器運(yùn)行中本身拋出的異常,以及調(diào)用模型的業(yè)務(wù)方法的異常. 當(dāng)異常被控制器捕獲的時(shí)候,在異常處理代碼塊中,創(chuàng)建描述異常信息的ActionMessage對(duì)象把它保存在ActionMessages或者它的子類 ActionErrors對(duì)象中,然后把它保存在特定的范圍內(nèi),比如request或session.接下來在視圖層 標(biāo)簽檢索特定范圍的ActionMessages對(duì)象,把本地化錯(cuò)誤消息輸出到頁(yè)面上.這種異常處理機(jī)制可以避免用戶看到原始的 java異常信息.可以更友好的把錯(cuò)誤信息展示給用戶. Struts 框架處理異常是以JVM的異常處理機(jī)制為基礎(chǔ)的,盡管它提供了強(qiáng)大的通用錯(cuò)誤處理機(jī)制,但是不能保證捕獲到所有的異常或者錯(cuò)誤.當(dāng)錯(cuò)誤發(fā)生的時(shí)候,如果 Struts不能處理這種異常或者錯(cuò)誤,就把錯(cuò)誤拋給JavaWeb容器.容器先查看是否在Web應(yīng)用發(fā)布描述文件中配置了元素,如果存在該元素就返回元素的子元素指定的錯(cuò)誤頁(yè)面,否則就會(huì)把錯(cuò)誤直接拋給用戶.下面簡(jiǎn)單介紹一 Struts幾個(gè)重要的類的異常處理機(jī)制,如果想了解這幾個(gè)類的更深的異常機(jī)制,建議查看Struts源代碼. 1) ActionServlet類的peocess()方法不捕獲任何異常,僅僅聲明向上層調(diào)用方法拋出異常. 2) RequestProcessor類是Struts框架處理異常的核心組件. 3) ExceptionHandler類是默認(rèn)的異常處理類,它的execute()方法負(fù)責(zé)處理異常. 根據(jù)打造技術(shù)Blog,尋求JAVA精髓! Servlet規(guī)范,當(dāng)容器捕獲到異常的時(shí)候,將查看是否在Web.xml中配置了相應(yīng)的元素,如果存在,就會(huì)返回其子元素指定的錯(cuò)誤頁(yè)面.舉個(gè)簡(jiǎn)單明了的例子如下: 500 /error.jsp javax.servlet.ServletException /error.jsp Struts框架也允許以配置的方式來處理異 常,配置方法可以避免在Action類中通過硬編碼來處理異常,提高應(yīng)用的靈活性,可重用性和可維護(hù)性.對(duì)于Action類的execute()方法拋出 的異常會(huì)先查找異常處理元素,如果是被嵌套在元素中就表示是全局異常處理元素,對(duì)所有的Action都適用.如果是嵌套在元素中,就表示局部的異常處理元素,僅對(duì)當(dāng)前的Action適用.闡 述一下元素的屬性: 1) type:指定待處理的異常類 2) handler:指定異常處理類.默認(rèn)是ExceptionHandler.如果自己要定義必須繼承它 3) path:指定轉(zhuǎn)發(fā)路徑 4) key:指定錯(cuò)誤消息key.根據(jù)這個(gè)key到ResourceBundle中尋找匹配的消息文本 5) bundle:指定ResourceBundle,如果沒設(shè)置將使用哦么人的ResourceBundle 6) scope:指定ActionMessages對(duì)象的存放范圍,默認(rèn)request,也可以設(shè)置session Struts框架提供了強(qiáng)大的異常處理功能,Struts控制器負(fù)責(zé)捕獲異常,并把異常包裝與 ResourceBundle綁定的 ActionMessages對(duì)象,在視圖層 標(biāo)簽?zāi)軌蝻@示出來,主要可以通過配置的方式和編程的方式實(shí)現(xiàn),這里不推 薦用編程的方法,可以增強(qiáng)程序的靈活性,可重用性,可維護(hù)性. (網(wǎng)友們的支持,是我繼續(xù)寫技術(shù)文章的動(dòng)力!) Struts的異常處理機(jī)制總的來說,在struts新的版本中加入了對(duì)異常的處理,稱之為:Exception Handling,標(biāo)志著作為一個(gè)整體的解決框架,struts原來越趨于成熟。 通 常來說,以前在用struts開發(fā)的過程中,對(duì)于異常的處理,主要是采用手動(dòng)處理的方式:如通過try/catch等等捕獲異常,然后定制個(gè)性化的比較詳 細(xì)的錯(cuò)誤信息放進(jìn)ActionError中,然后在具體的返回頁(yè)面中把這些錯(cuò)誤信息反饋給用戶(包括開發(fā)員)。異常原始的信息不管是最終用戶還是開發(fā)員都是不希望看到的。 下面著重講一下在struts中是如何通過配置文件來解決異常。 Struts中的Exception Handleing不難,簡(jiǎn)單高效是業(yè)內(nèi)給其的一個(gè)比較好的評(píng)價(jià)。 通過配置文件(主要是struts-config.xml)來定制異常處理,就象定義formbean一樣,定制異常也有兩種方法,姑且把它分為:“全局異常”和“局部異常”。 全局異常,定義方法如下: ………… ………… 上 述代碼在struts-config.xml中定義了一個(gè)全局異常,它的作用是拋出InvalidiItemsCatalogNameException (本處的意思是當(dāng)在添加商品分類的時(shí)候發(fā)現(xiàn)該類別已經(jīng)存在)異常的時(shí)候返回到error.jsp中,并且攜帶自定的比較規(guī)范的異常信息 expired.InvalidItemsCatalogName,expired.InvalidItemsCatalogName可以在應(yīng)用程序的資源配置文件中找到,如: expired.InvalidItemsCatalogName=你要添加的商品類別已經(jīng)存在,請(qǐng)?zhí)砑有碌念悇e! 局部異常,定義方法如下: ………… ………… (關(guān)于Tiles的內(nèi)容參看我的另一篇文章“Struts使用Tiles輔助開發(fā)”) 下面我們把關(guān)注的目光放在具體action里,看看struts是如何進(jìn)行異常處理的 //ItemsCatalogAction.java package com.iplateau.jshop.action.ItemsCatalogAction import com.iplateau.jshop.business.ItemsCatalogMap; import com.iplateau.jshop.action.ItemsCatalogForm; import ***; public class ItemsCatalogAction extends BaseAction { public ActionForward execute( ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response) throws Exception { String flg=”error”; String act=request.getParameter(“act”); ItemsCatalogMap map=new ItemsCatalogMap(); ItemsCatalogForm thisform=( ItemsCatalogForm)form; // ItemsCatalogMap為具體的處理商品類別的類 if(act.equals(“create”)) { map.validateNameOfCatalog(thisform); map.create(thisform); flg=”success”; return mapping.findForward(flg); } else{ return mapping.findForward(flg); } } } 上便是一個(gè)簡(jiǎn)單的action,其中用act的具體內(nèi)容來判斷此時(shí)的action要處理的操作,如act為cteate的時(shí)候,處理的是商品類別的添加等等。 看了上邊的代碼,可以了解,在調(diào)用具體的業(yè)務(wù)類進(jìn)行添加商品類別以前先要判斷此時(shí)要添加的商品類別的合法性,如: 調(diào)用map.validateNameOfCatalog(thisform); 其中在ItemsCatalogMap定義的validateNameOfCatalog 方法拋出InvalidiItemsCatalogNameException。 下面讓我們運(yùn)行一下添加商品類別的例子,如果此時(shí)數(shù)據(jù)庫(kù)中已經(jīng)存在“牛奶制品”這個(gè)商品類別,而我們又要添加此類別的時(shí)候,程序如我們預(yù)想的一樣轉(zhuǎn)到error.jsp并且顯示了我們要先是的錯(cuò)誤信息:你要添加的商品類別已經(jīng)存在,請(qǐng)?zhí)砑有碌念悇e! 其實(shí)就這么簡(jiǎn)單,我們的定制的異常在程序中并不需要用try/catch來捕獲,一旦出現(xiàn)了我們已經(jīng)定義的異常那么就會(huì)轉(zhuǎn)到相應(yīng)得頁(yè)面,并且攜帶定制的信息。 還記得在struts先前的版本中我們的請(qǐng)求都是通過action的perform來處理,可是現(xiàn)在都要通過execute來處理,就其原因一個(gè)很重要的就是“成全”Exception Handling。為什么呢? 因?yàn)閜erform在聲明的時(shí)候僅僅拋出IOException 和ServletException,這遠(yuǎn)遠(yuǎn)不能滿足Exception Handling的要求,那么讓我們看看execute是怎樣的:它籠統(tǒng)的拋出Exception(所有異常的父類) 上 面的闡述只是一個(gè)引子,而且默認(rèn)struts的異常是通過org.apache.struts.action. ExceptionHandler來處理的,你可以定義自己的處理方式,只要繼承它并實(shí)現(xiàn)其中的execute方法,這個(gè)方法在ExceptionHandler的定義如下: public ActionForward execute (Exception ex, ExceptionConfig ae, ActionMapping mapping, ActionForm formInstance, HttpServletRequest request, HttpServletResponse response) throws ServletException 具體而且專業(yè)的分析Exception Handling的內(nèi)容,請(qǐng)參看《Programming Jakarta Struts》第10章??這本書的電子版網(wǎng)上很多地方可以下載。 原文地址 http://tech.ccidnet.com/pub/article/c322_a74029_p1.html