Java Web项目整体异常处理机制

上一篇 / 下一篇  2012-09-17 15:24:12 / 个人分类:Java

'e;j4N+Y2Qd0  在实际的j2ee项目中,系统内部难免会出现一些异常,如果把异常放任不管直接打印到浏览器可能会让用户感觉莫名其妙,也有可能让某些用户找到破解系统的方法。51Testing软件测试网,^.D0oL ~lY*V,g$\

6a G {N1^QnU9T0  出来工作一年时间了,我也大概对异常处理有了一些了解,在这呢小弟简单介绍下个人对异常处理的见解,抛砖引玉,希望各位大神提出宝贵的意见和建议。

D7I[3ytc051Testing软件测试网S5` P#D6Ro

   就拿spring+struts2+hibernate项目说明:通常一个页面请求到后台以后,首先是到action(也就是所谓mvc的 controller),在action层会调用业务逻辑service,servce层会调用持久层dao获取数据。最后执行结果会汇总到 action,然后通过action控制转发到指定页面,执行流程如下图所示:51Testing软件测试网#t^]| c qMm

51Testing软件测试网:u8[ M/[F]O#csi

   而这三层其实都有可能发生异常,比如dao层可能会有SQLException,service可能会有 NullPointException,action可能会有IOException,一但发生异常并且程序员未做处理,那么该层不会再往下执行,而是向 调用自己的方法抛出异常,如果dao、service、action层都未处理异常的话,异常信息会抛到服务器,然后服务器会把异常直接打印到页面,结果 就会如下图所示:

j^%R K-O8ix0

0`Cz X?n pOc^0

  其实这种错误对于客户来说毫无意义,因为他们通常是看不懂这是什么意思的。51Testing软件测试网6s$y4Z%Z v,P

  刚学java的 时候,我们处理异常通常两种方法:①直接throws,放任不管;②写try...catch,在catch块中不作任何操作,或者仅仅 printStackTrace()把异常打印到控制台。第一种方法最后就造就了上图的结果;而第二种方法更杯具:页面不报错,但是也不执行用户的请求, 简单的说,其实这就是bug(委婉点:通常是这样)!51Testing软件测试网.l}|,p,B D+f*]4D

  那么发生异常到底应该怎么办呢?我想在大家对java异常有一定了解以后,会知道:异常应该在action控制转发之前尽量处理,同时记录log日志,然后在页面以友好的错误提示告诉用户出错了。大家看下面的代码:51Testing软件测试网9s4~&[uv)xu

//创建日志对象
!Q!_M^N0Log log = LogFactory.getLog(this.getClass());51Testing软件测试网(Nwbu1~E
 51Testing软件测试网rB'n/D(BX.yz&L(Qc
//action层执行数据添加操作51Testing软件测试网|9lJ jy I
public String save(){
i?1Z%_4kQ RZ,OZ0   try{
{u c O]!c n"{!}0         //调用service的save方法51Testing软件测试网y BLz}*M1l2i@C
         service.save(obj);51Testing软件测试网~#L v0]p[-~+K m
   }catch(Exception e){51Testing软件测试网Vg*w/`s2K&e1Ygb
         log.error(...);   //记录log日志
+\#g+j/A m0      return "error"; 到指定error页面
\*H$C yD;V2Q7[B0s0   }51Testing软件测试网G?$a1g~5{G4E
   return "success";51Testing软件测试网dZ_vI BPC2j`A
}

  如果按照上面的方式处理异常以后,我们用户最后看到的页面可能就会是下面这种形式(我想这种错误提示应该稍微友好点了吧):

NW/j*IoY0

C3Wp2w&Y J O/\0

  然后我们回到刚才处理异常的地方,如果大家积累了一些项目经验以后会发现使用上面那种处理异常的方式可能还不够灵活:51Testing软件测试网~ Jo8~xJ2O1X&l

  ①因为spring把大多数非运行时异常都转换成运行时异常(RuntimeException)最后导致程序员根本不知道什么地方应该进行try...catch操作

.s%W;CH zLX0

  ②每个方法都重复写try...catch,而且catch块内的代码都很相似,这明显做了很多重复工作而且还很容易出错,同时也加大了单元测试的用例数(项目经理通常喜欢根据代码行来估算UT case)51Testing软件测试网7_v"^"a,w'[r1~

  ③发生异常有很多种情况:可能有数据库增删改查错误,可能是文件读写错误,等等。用户觉得每次发生异常都是“访问过程中产生错误,请重试”的提示完全不能说明错误情况,他们希望让异常信息更详尽些,比如:在执行数据删除时发生错误,这样他们可以更准确地给维护人员提供bug信息。51Testing软件测试网&G*q|3D(MIWr8l

51Testing软件测试网9i8Z*L N8MX0bE-f5N\

  如何解决上面的问题呢?我是这样做的:JDK异常或自定义异常+异常拦截器

Df_#USJ$h0

"v?*Q3k ]6A{"Ux0  struts2拦截器的作用在网上有很多资料,在此不再赘述,我的异常拦截器原理如下图所示:51Testing软件测试网@r)eRv8Z%N

/Z m `EQ5e0

  首先我的action类、service类和dao类如果有必要捕获异常,我都会try...catch,catch块内不记录log,通常是抛出一个新异常,并且注明错误信息:

!og'A5ZyO['k n0

51Testing软件测试网bpz m N.w` p7J

//action层执行数据添加操作51Testing软件测试网ffwW7GclO
public String save(){
`x$T@5~!Co_%d0   try{51Testing软件测试网 RSLQQAB!o.v
         //调用service的save方法51Testing软件测试网/k)T2g4a6d
         service.save(obj);51Testing软件测试网T l*t+|J
   }catch(Exception e){
d!X+N sHUAK)B/\k0      //你问我为什么抛出Runtime异常?因为我懒得在方法后写throws  xx
K"VJ~j)v:EA&v0      throw new RuntimeException("添加数据时发生错误!",e);
9}L5jBD*C h0  }51Testing软件测试网s"Z$_-@plQ
   return "success";
(J1}'J%zf2f.fF7Zl M0}

  然后在异常拦截器对异常进行处理,看下面的代码:

!Qc5~[+~aO{.T!m0

51Testing软件测试网*JA]*]_ E R

public String intercept(ActionInvocation actioninvocation) {51Testing软件测试网[Fzz9TK rP
 51Testing软件测试网"B2c/q0Z&HS+p3c e
  String result = null; // Action的返回值
e^/no|S)]{G2}J y0  try {
Th1n-p @IHV#b0   // 运行被拦截的Action,期间如果发生异常会被catch住51Testing软件测试网 M.lV{%hT&\X$D
   result = actioninvocation.invoke();
)?1RoP'a4_Ut0   return result;51Testing软件测试网-q#W mN&x)|[i
  } catch (Exception e) {
D8?iw,r_ u0   /**
u-ar1kZ+E#n'_p0I0    * 处理异常51Testing软件测试网?2o$V kL0F+Fjp$vmW
    */51Testing软件测试网,a3x ejai/t)`9a2I
   String errorMsg = "未知错误!";
d!\d+x K8F E Sr0   //通过instanceof判断到底是什么异常类型51Testing软件测试网c)?V,_] Y"{c
   if (e instanceof BaseException) {
,rZ NV!H0    BaseException be = (BaseException) e;51Testing软件测试网5G;ZhG2E+k0]tSDX6n
    be.printStackTrace(); //开发时打印异常信息,方便调试
:e8E7Q9KF"GJa^][0    if(be.getMessage()!=null||Constants.BLANK.equals(be.getMessage().trim())){51Testing软件测试网sr b6E1|&i6^
     //获得错误信息
P|)Y*HRLW0     errorMsg = be.getMessage().trim();51Testing软件测试网 dE!aI.L o-?
    }51Testing软件测试网i/o E,O#F7O
   } else if(e instanceof RuntimeException){
&?;h)v"q5M m'b y0    //未知的运行时异常
7K5e,P d{_D {0    RuntimeException re = (RuntimeException)e;51Testing软件测试网2?GB}8k]@[Rxu
    re.printStackTrace();
?*O.A4I?(I/Mn,U6\p0   } else{
*ba"MTaB2~0    //未知的严重异常
#T7H,|/@P&Lq'MP0    e.printStackTrace();
s~Xb6l^r{0   }
I:b5{3yX0   //把自定义错误信息51Testing软件测试网M4{0|I yF\3R
   HttpServletRequest request = (HttpServletRequest) actioninvocation51Testing软件测试网 G`jg3l
     .getInvocationContext().get(StrutsStatics.HTTP_REQUEST);51Testing软件测试网t1^-tgZ
   51Testing软件测试网2A&^q#o1R$A
   /**51Testing软件测试网|[-K G(zb
    * 发送错误消息到页面
D p L K MyZ"`0    */51Testing软件测试网9U7g1W le!z
   request.setAttribute("errorMsg", errorMsg);
(g'oBr,L({0  
IV(KvVG]K0   /**51Testing软件测试网0dG d qcNr3_
    * log4j记录日志
6JH r&xz3\a0    */51Testing软件测试网Nj |cy W:s^
   Log log = LogFactory
G/Y.a#X~\4g0     .getLog(actioninvocation.getAction().getClass());51Testing软件测试网6a7?!}:D3p*`#v
   if (e.getCause() != null){
R6[-PJ*I S0    log.error(errorMsg, e);
f#e"ocQ^?0   }else{51Testing软件测试网+v7s }'[uZP1Kv:?
    log.error(errorMsg, e);51Testing软件测试网P,UP2S#x1_A
   }51Testing软件测试网xk+}?+PM(Z\s,z
 51Testing软件测试网%[E#n+tj3_0J
   return "error";51Testing软件测试网U0H*E5\'e8Qo
  }// ...end of catch
j5zj%XVl0 }
#SX-\ y5|$LE-h0 需要注意的是:在使用instanceof判断异常类型的时候一定要从子到父依次找,比如BaseException继承与RuntimeException,则必须首先判断是否是BaseException再判断是否是RuntimeException。
51Testing软件测试网.hT7P:gqj9I

  最后在error JSP页面显示具体的错误消息即可:51Testing软件测试网-fBe&?7~T"N

51Testing软件测试网!Uw@Y/Yo i.x@

} Y-Pk3U0TS0
<body>51Testing软件测试网t b4JQ Jx
<s:if test="%{#request.errorMsg==null}">
&B}L,jwS)Sh0 <p>对不起,系统发生了未知的错误</p>
EW4p?;O:k.s!W0</s:if>51Testing软件测试网u)zVM,{5nV-E
<s:else>
H&m ?5y)z+Pu0 <p>${requestScope.errorMsg}</p>
m0X}:HT [0</s:else>51Testing软件测试网W%Qz'tk.X-q*R"j$hX']
</body>
51Testing软件测试网#Q0U*m%K@2]%s5O

  以上方式可以拦截后台代码所有的异常,但如果出现数据库连接异常时不能被捕获的,大家可以使用struts2的全局异常处理机制来处理:

)~3S F,q+LC z051Testing软件测试网R:iN[L?'t

z:iH6lw0E+u0
<global-results>51Testing软件测试网4wK Ju1u4Y2e F
 <result name="error" >/Web/common/page/error.jsp</result>51Testing软件测试网B!GT7vg2h,y
</global-results>51Testing软件测试网9{+J2[ Zy&i/o-v2Y
  
S@J)q}k7p2Y0<global-exception-mappings>51Testing软件测试网&E"Z(Pw9jj-E
 <exception-mapping result="error" exception="java.lang.Exception"></exception-mapping>51Testing软件测试网alv'?!J:dM7b#t
</global-exception-mappings>
51Testing软件测试网t]5^l3V/UR#d"T

  上面这是一个很简单的异常拦截器,大家可以使用自定义异常,那样会更灵活一些。51Testing软件测试网 hB.f:]S;s*W;nw

51Testing软件测试网'M:roS,HC W

  以上异常拦截器可以使用其它很多技术替换:比如spring aop,servlet filter等,根据项目实际情况处理。51Testing软件测试网"P0[+| Jl+p7|

51Testing软件测试网Mz StP)AD@+O{ ]

  【补充】ajax也可以进行拦截,但是因为ajax属于异步操作,action通过response形式直接把数据返回给ajax回调函数,如果发生异常,ajax是不会执行页面跳转的,所以必须把错误信息返回给回调函数,我针对json数据的ajax是这样做的:

,c\0O&oPz*a0

(M-Pbd+^@2Ez0

t:y?'j'}v0
   /**
8r#c NQ v{%X i8j0    * 读取文件,获取对应错误消息51Testing软件测试网 Z5ztZ5}[
    */51Testing软件测试网1L0^1Xey-e
   HttpServletResponse response = (HttpServletResponse)actioninvocation.getInvocationContext().get(StrutsStatics.HTTP_RESPONSE);51Testing软件测试网}B1H.Uu'I
   response.setCharacterEncoding(Constants.ENCODING_UTF8);
HOe:H0u2CY3e0   /**
x,h&V*q:@L$gw0    * 发送错误消息到页面
wU]'p[0    */51Testing软件测试网,G;cCSM
   PrintWriter out;
.^x1h/as1~&M0   try {51Testing软件测试网H:s'\KB%S&nL
    out = response.getWriter();
it,u7xQF&xw H0    Message msg = new Message(errorMsg);51Testing软件测试网PK3wy IfxD
    //把异常信息转换成json格式返回给前台51Testing软件测试网;NfXy b
    out.print(JSONObject.fromObject(msg).toString());
V,SzS;S3W*S0   } catch (IOException e1) {
9r"E8^8s h4l4c#z0    throw e;51Testing软件测试网C8vJ"u*O&dW\|&H?
   }
51Testing软件测试网O F$C xz5[B

  以上是个人拙见,勿拍砖,谢谢。51Testing软件测试网1iZ O U}q


o i2])| tR]'B0

TAG:

 

评分:0

我来说两句

Open Toolbar