`
寻觅一只耳朵
  • 浏览: 35522 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

j2ee异常拦截

    博客分类:
  • java
阅读更多

原文出自:IT博客吧(http://www.itblog8.cn/java/20130530133.html

在实际的j2ee项目中,系统内部难免会出现一些异常,如果把异常放任不管直接打印到浏览器可能会让用户感觉莫名其妙,也有可能让某些用户找到破解系统的方法。

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

 

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

 

 

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

 

 

  其实这种错误对于客户来说毫无意义,因为他们通常是看不懂这是什么意思的。

 

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

 

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

 

//创建日志对象
Log log =
LogFactory.getLog(this.getClass());

//action层执行数据添加操作
public String
save(){
try{
//调用service的save方法

service.save(obj);
}catch(Exception e){
log.error(...);
//记录log日志
return "error"; 到指定error页面
}
return
"success";
}

 

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

 

 

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

 

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

 

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

 

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

 


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


  struts2拦截器的作用在网上有很多资料,在此不再赘述,我的异常拦截器原理如下图所示:



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



//action层执行数据添加操作
public String save(){
try{

//调用service的save方法
service.save(obj);
}catch(Exception
e){
//你问我为什么抛出Runtime异常?因为我懒得在方法后写throws xx
throw new
RuntimeException("添加数据时发生错误!",e);
}
return
"success";
}

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


public String intercept(ActionInvocation actioninvocation)
{

String result = null; // Action的返回值
try {
//
运行被拦截的Action,期间如果发生异常会被catch住
result =
actioninvocation.invoke();
return result;
} catch (Exception e)
{
/**
* 处理异常
*/
String errorMsg =
"未知错误!";
//通过instanceof判断到底是什么异常类型
if (e instanceof BaseException)
{
BaseException be = (BaseException) e;
be.printStackTrace();
//开发时打印异常信息,方便调试
if(be.getMessage()!=null||Constants.BLANK.equals(be.getMessage().trim())){
//获得错误信息
errorMsg
= be.getMessage().trim();
}
} else if(e instanceof
RuntimeException){
//未知的运行时异常
RuntimeException re =
(RuntimeException)e;
re.printStackTrace();
}
else{
//未知的严重异常
e.printStackTrace();
}
//把自定义错误信息
HttpServletRequest
request = (HttpServletRequest)
actioninvocation
.getInvocationContext().get(StrutsStatics.HTTP_REQUEST);

/**

* 发送错误消息到页面
*/
request.setAttribute("errorMsg",
errorMsg);

/**
* log4j记录日志
*/
Log log =
LogFactory
.getLog(actioninvocation.getAction().getClass());
if
(e.getCause() != null){
log.error(errorMsg,
e);
}else{
log.error(errorMsg, e);
}

return
"error";
}// ...end of catch
}

 


  需要注意的是:在使用instanceof判断异常类型的时候一定要从子到父依次找,比如BaseException继承与RuntimeException,则必须首先判断是否是BaseException再判断是否是RuntimeException。


  最后在error JSP页面显示具体的错误消息即可:



<body>
<s:if
test="%{#request.errorMsg==null}">
<p>对不起,系统发生了未知的错误</p>
</s:if>
<s:else>
<p>${requestScope.errorMsg}</p>
</s:else>
</body>

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



<global-results>
<result name="error"
>/Web/common/page/error.jsp</result>
</global-results>


<global-exception-mappings>
<exception-mapping result="error"
exception="java.lang.Exception"></exception-mapping>
</global-exception-mappings>

  上面这是一个很简单的异常拦截器,大家可以使用自定义异常,那样会更灵活一些。


  以上异常拦截器可以使用其它很多技术替换:比如spring aop,servlet filter等,根据项目实际情况处理。


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



/**
* 读取文件,获取对应错误消息
*/
HttpServletResponse response =
(HttpServletResponse)actioninvocation.getInvocationContext().get(StrutsStatics.HTTP_RESPONSE);
response.setCharacterEncoding(Constants.ENCODING_UTF8);
/**

* 发送错误消息到页面
*/
PrintWriter out;
try {
out =
response.getWriter();
Message msg = new
Message(errorMsg);
//把异常信息转换成json格式返回给前台
out.print(JSONObject.fromObject(msg).toString());
}
catch (IOException e1) {
throw e;
}

 

分享到:
评论

相关推荐

    J2EE项目异常处理的几种方案

    例如,定义基类`BaseException`,并根据业务需求扩展多个子类,如`BusinessException`、`SystemException`等,以便于在异常拦截器中进行更细致的异常类型判断和处理。 ```java public abstract class BaseException...

    J2EE网上购物系统

    1. **Struts2**:Struts2是一个强大的MVC框架,通过拦截器机制处理HTTP请求,提供了一系列的拦截器来实现如认证、授权、异常处理等功能。它的Action类负责接收并处理请求,然后返回结果到视图。Struts2还支持多种...

    J2EE考试复习材料

    过滤器是J2EE应用程序中的一个组件,用于拦截和处理用户请求。我们可以开发一个过滤器来显示一个JSP页面的执行时间。 十二、JavaBean DatabaseBean JavaBean DatabaseBean是一个简单的JavaBean,用于连接数据库并...

    j2ee相关试题 j2ee相关试题

    `exception`对象(B选项)在JSP中用于处理运行时发生的异常情况,可以通过它获取异常的具体信息。 ### 17. 内置对象 最后,题目中的示例代码展示了JSP内置对象的使用,具体涉及到哪些内置对象需要根据代码进一步分析...

    j2ee 1.6 帮助文档

    1. **基础类库**:如Object、String、List、Map等基本数据结构,以及异常处理、输入输出流、多线程、网络编程等。 2. **集合框架**:HashMap、ArrayList、LinkedList等,以及泛型的引入,增强了类型安全性和代码...

    J2EE架构师推荐书目录

    - **重点内容**:MVC模式、拦截器模式、工厂模式等在J2EE项目中的具体应用。 - **EJB设计模式**:针对EJB开发过程中的常见问题提供解决方案。 - **重点内容**:EJB生命周期管理、状态模式的应用等。 ### 开源框架...

    j2ee真的认证复习完整版

    - **解析**:**Filter** 是一种特殊的Servlet,用于拦截客户端请求,在请求到达目标资源之前进行预处理,或者在响应返回客户端之前进行后处理。它可以用于执行各种任务,如身份验证、日志记录等。 ### 11. 实现 ...

    简述J2EE的核心技术

    3. **强大的拦截器**:Struts2的拦截器机制可以方便地实现如权限验证、日志记录等功能。 4. **易于测试**:Struts2支持单元测试和集成测试,有利于提升代码质量。 5. **插件机制**:丰富的插件支持,可以快速扩展...

    j2ee控制器及应用

    综上所述,J2EE中的控制器在Web应用程序中起着至关重要的作用,而在Struts框架中,Action和拦截器共同构成了强大的控制器机制。深入理解和实践这些概念,对于提升Java Web开发技能至关重要。通过研究提供的Struts...

    j2ee合集

    而"struts2.pdf"可能是一个全面的教程,涵盖Struts2的基础到高级特性,包括但不限于Action、拦截器(Interceptor)、结果类型(Result)、标签库、国际化(i18n)和异常处理。 综上所述,这个"j2ee合集"是学习和...

    j2ee人事管理系统

    通过阅读和理解代码,我们可以学习到如何配置和使用SpringMVC的拦截器、异常处理器,以及如何进行数据验证。同时,对于前端展示,可能涉及到JSP、FreeMarker或者Thymeleaf等模板引擎,它们负责将后端数据渲染成用户...

    j2EE 经典面试题

    - **拦截器(Interceptor)**:了解Struts2的拦截器机制,实现自定义拦截逻辑。 - **OGNL表达式**:使用OGNL进行数据绑定和传递。 9. **Web服务** - **SOAP与RESTful**:理解两种Web服务的区别,以及如何使用JAX...

    j2ee开发人员机试题

    在实际的机试题中,可能会涉及到SSH框架的配置、异常处理、事务管理、数据绑定、模型驱动等具体知识点。例如,要求考生配置Struts2的Action拦截器,编写Spring的Bean配置,或者使用Hibernate进行数据库操作。了解和...

    J2EE框架学习笔记

    这包括了解如何配置和使用框架,以及如何处理它们之间的协作问题,如异常处理、事务管理等。此外,对于现代开发环境,还需要关注框架的最新发展,如Spring Boot的自动配置和微服务支持,以及JDBC的替代方案如JPA...

    J2EE1.5.rar

    Filter则允许开发者在请求到达Servlet之前或之后进行拦截和处理,例如数据过滤、安全检查等。 接着,javax.servlet.http 包扩展了 javax.servlet,包含了HttpServlet类和HttpSession接口。HttpServlet是Servlet的...

    J2EE企业级项目开发-3期(KC007) 8.3 Struts之拦截器使用技巧考试.doc

    - f) `exception`拦截器:处理未捕获的异常,可以自定义异常处理逻辑。 - g) `validation`拦截器:根据Action的注解或配置文件进行数据验证。 - h) `token`拦截器:防止重复提交。 - i) `conversionError`拦截器...

    java笔记 Java-Web笔记 J2EE三大框架笔记

    过滤器和监听器作为扩展点,可以实现请求拦截、会话管理等功能,增强了应用的灵活性和可维护性。 接下来,J2EE(Java Platform, Enterprise Edition)是Java在企业级应用开发中的标准平台,它定义了一系列规范和...

    j2EE面试题分类集合

    - **异常处理**:了解try-catch-finally语句块,异常类层次结构以及自定义异常。 - **集合框架**:List、Set、Map接口的实现类如ArrayList、HashSet、HashMap的特性和使用。 - **多线程**:线程的创建方式、同步...

    两个j2ee网上商城项目源码

    它支持拦截器,可以添加自定义行为,如验证和异常处理。 3. **Hibernate ORM**:Hibernate简化了数据库操作,通过实体类与数据库表进行映射,使得开发者可以使用对象而非SQL语句进行数据操作。它还支持事务管理和...

    基于springmvc的J2EE平台

    在基于SpringMVC的J2EE平台上,我们将深入探讨以下几个关键知识点: 1. **SpringMVC架构**: - **模型(Model)**:模型层通常由JavaBean或领域对象组成,负责存储和处理业务数据。 - **视图(View)**:视图层...

Global site tag (gtag.js) - Google Analytics