`

java web项目整体异常处理机制 .

 
阅读更多

来源:http://blog.csdn.net/luqin1988/article/details/7970988

 

在实际的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日志,然后在页面以友好的错误提示告诉用户出错了。大家看下面的代码:

  1. //创建日志对象   
  2. Log log = LogFactory.getLog(this.getClass());  
  3.    
  4. //action层执行数据添加操作   
  5. public String save(){  
  6.    try{  
  7.          //调用service的save方法   
  8.          service.save(obj);  
  9.    }catch(Exception e){  
  10.          log.error(...);   //记录log日志   
  11.       return "error"; 到指定error页面  
  12.    }  
  13.    return "success";  
  14. }  
//创建日志对象
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,通常是抛出一个新异常,并且注明错误信息:

  1. //action层执行数据添加操作   
  2. public String save(){  
  3.    try{  
  4.          //调用service的save方法   
  5.          service.save(obj);  
  6.    }catch(Exception e){  
  7.       //你问我为什么抛出Runtime异常?因为我懒得在方法后写throws  xx   
  8.       throw new RuntimeException("添加数据时发生错误!",e);  
  9.   }  
  10.    return "success";  
  11. }  
//action层执行数据添加操作
public String save(){
   try{
         //调用service的save方法
         service.save(obj);
   }catch(Exception e){
      //你问我为什么抛出Runtime异常?因为我懒得在方法后写throws  xx
      throw new RuntimeException("添加数据时发生错误!",e);
  }
   return "success";
}

 

 

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

  1. public String intercept(ActionInvocation actioninvocation) {  
  2.    
  3.         String result = null// Action的返回值   
  4.         try {  
  5.             // 运行被拦截的Action,期间如果发生异常会被catch住   
  6.             result = actioninvocation.invoke();  
  7.             return result;  
  8.         } catch (Exception e) {  
  9.             /** 
  10.              * 处理异常 
  11.              */  
  12.             String errorMsg = "未知错误!";  
  13.             //通过instanceof判断到底是什么异常类型   
  14.             if (e instanceof BaseException) {  
  15.                 BaseException be = (BaseException) e;  
  16.                 be.printStackTrace(); //开发时打印异常信息,方便调试   
  17.                 if(be.getMessage()!=null||Constants.BLANK.equals(be.getMessage().trim())){  
  18.                     //获得错误信息   
  19.                     errorMsg = be.getMessage().trim();  
  20.                 }  
  21.             } else if(e instanceof RuntimeException){  
  22.                 //未知的运行时异常   
  23.                 RuntimeException re = (RuntimeException)e;  
  24.                 re.printStackTrace();  
  25.             } else{  
  26.                 //未知的严重异常   
  27.                 e.printStackTrace();  
  28.             }  
  29.             //把自定义错误信息   
  30.             HttpServletRequest request = (HttpServletRequest) actioninvocation  
  31.                     .getInvocationContext().get(StrutsStatics.HTTP_REQUEST);  
  32.               
  33.             /** 
  34.              * 发送错误消息到页面 
  35.              */  
  36.             request.setAttribute("errorMsg", errorMsg);  
  37.           
  38.             /** 
  39.              * log4j记录日志 
  40.              */  
  41.             Log log = LogFactory  
  42.                     .getLog(actioninvocation.getAction().getClass());  
  43.             if (e.getCause() != null){  
  44.                 log.error(errorMsg, e);  
  45.             }else{  
  46.                 log.error(errorMsg, e);  
  47.             }  
  48.    
  49.             return "error";  
  50.         }// ...end of catch   
  51.     }  
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页面显示具体的错误消息即可:

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

 

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

  1. <global-results>  
  2.     <result name="error" >/Web/common/page/error.jsp</result>  
  3. </global-results>  
  4.            
  5. <global-exception-mappings>  
  6.     <exception-mapping result="error" exception="java.lang.Exception"></exception-mapping>  
  7. </global-exception-mappings>  
<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是这样做的:

  1. /** 
  2.  * 读取文件,获取对应错误消息 
  3.  */  
  4. HttpServletResponse response = (HttpServletResponse)actioninvocation.getInvocationContext().get(StrutsStatics.HTTP_RESPONSE);  
  5. response.setCharacterEncoding(Constants.ENCODING_UTF8);  
  6. /** 
  7.  * 发送错误消息到页面 
  8.  */  
  9. PrintWriter out;  
  10. try {  
  11.     out = response.getWriter();  
  12.     Message msg = new Message(errorMsg);  
  13.     //把异常信息转换成json格式返回给前台   
  14.     out.print(JSONObject.fromObject(msg).toString());  
  15. catch (IOException e1) {  
  16.     throw e;  
  17. }  
分享到:
评论

相关推荐

    java web项目整体异常处理机制.doc

    当我们谈论Java Web项目的整体异常处理机制时,我们关注的是如何优雅地处理可能出现的各种异常,避免将错误信息直接暴露给用户,同时确保系统能够进行必要的错误记录和反馈。 在Spring+Struts2+Hibernate这样的经典...

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

    总结起来,Java Web项目的整体异常处理机制应包括以下几个关键点: 1. 在Action、Service和DAO层使用try-catch块,捕获异常并抛出自定义异常。 2. 使用日志工具记录异常信息,便于排查问题。 3. 设计并实现异常拦截...

    Java Web应用框架研究与应用.pdf

    在实际的Java Web应用开发中,Struts2常用于处理用户请求和展示视图,Hibernate负责数据的持久化,而Spring则作为一个整体的协调者,管理和连接其他两个框架,提供事务管理、安全控制等服务。这种组合使得开发团队能...

    《java学习》-Java web开发async机制学习.zip

    Java Web开发中的异步机制是提高应用程序性能和响应能力的关键技术。在传统的Web开发中,...通过深入研究这个压缩包中的内容,你将全面了解Java Web开发中的异步处理机制,并具备在实际项目中应用这些知识的能力。

    简单JAVA WEB服务器

    这需要合理的异常处理机制,并可能涉及自定义错误页面的设置。 8. **安全性考虑**:虽然这是一个简单的项目,但安全性是任何Web服务器都应关注的。这包括但不限于输入验证、防止跨站脚本攻击(XSS)、SQL注入等。 ...

    精通struts基于mvc的java web设计与开发

    此外,孙卫琴还会介绍Struts的国际化支持、异常处理机制、Tiles框架用于页面布局,以及Struts与其他技术如Hibernate(ORM框架)或Spring(企业级应用框架)的整合。这些都是在实际开发中非常实用的知识点。 在...

    JavaMelody系统监控工具 v1.9.8.zip

    **集成JavaMelody到Java Web项目中:** 1. 添加依赖:将`javamelody-javamelody-core-1.98.0.jar`添加到项目的类路径中,如果是Maven项目,可以在pom.xml中加入对应的依赖项。 2. 配置初始化参数:在web.xml中添加...

    本科毕业设计项目-Java电影推荐系统-管理后台.zip

    1. **Java Web基础**:项目基于Java Web平台,利用Servlet和JSP进行服务器端编程。Servlet用于处理HTTP请求,而JSP则用于生成动态网页内容。理解这两者的关系和工作原理是构建Web应用的基础。 2. **MVC设计模式**:...

    Java web开发规范

    - **编码规范**:项目整体采用UTF-8编码,这是一种国际通用的字符编码标准,能支持全球多种语言,有利于项目的国际化。 3. **编码规范**: - **规则**:编码规范通常包括变量命名、注释编写、代码结构、错误处理...

    精通Struts基于MVC的Java Web设计与开发 孙卫琴 光盘

    第8章到第9章介绍了开发Struts应用的一些实用技术,如Struts框架的扩展点、Struts应用的国际化,Validator验证框架和异常处理机制等。第12章到第16章结合具体的Struts应用实例,详细介绍了Struts的标签库以及Tiles...

    Struts_In_Action使用领先的Java框架构建Web应用中文版.pdf

    本书还探讨了Struts框架的一些高级特性,例如自定义标签库、国际化支持、异常处理机制等。这些特性使得Struts框架更加灵活多变,能够满足不同场景下的需求。 #### 深入Struts架构 ##### 为什么我们需要Struts? ...

    基于Java的web网上书城方案设计书.docx

    - **安全性**:Java内置了多种安全机制,如沙箱模型和异常处理机制,确保程序的安全稳定运行。 - **Java在网络编程中的应用**: - Java支持通过URL访问网络资源,包括图像、音频、HTML文档和文本等。 - 可以定期...

    java web 开发 乐游网 概要设计

    【乐游网】Java Web开发概要设计文档旨在提供项目整体架构、主要功能模块和系统设计的概述,以便团队成员和利益相关者了解系统的基本结构和实现方式。以下是该概要设计的关键点: 1. **编写目的** - 本概要设计的...

    java_memcached-release_2.0.1.zip

    7. **异常处理**:客户端库会处理与服务器通信可能出现的各种异常,例如网络中断、超时等,并提供相应的异常类供开发者捕获和处理。 8. **配置与定制**:用户可以通过配置文件或代码设置连接参数,如超时时间、连接...

    基于java的电子相册管理系统源码.zip

    【标题】"基于Java的电子相册管理系统源码"是一个典型的软件开发项目,主要使用Java编程语言实现。这个系统可能是为了帮助用户管理个人照片、创建相册、分享相册以及进行其他与图片相关的操作。Java作为一种多平台的...

    基于java开发的极客学院小程序.zip

    12. **异常处理**: Java的异常处理机制能够帮助程序优雅地处理错误,通过try-catch-finally语句块捕获和处理异常。 以上知识点构成了基于Java开发小程序的基础,通过学习和实践,开发者可以创建出功能丰富、用户...

    java某在线教育后台管理系统.rar

    11. **异常处理和错误报告**:全局异常处理机制确保系统在遇到错误时能够优雅地处理,避免崩溃。错误日志和错误追踪系统如Sentry有助于快速定位并修复问题。 以上只是部分可能涉及的技术点,实际的在线教育后台管理...

    基于java的物流仓库管理系统源码.zip

    10. **异常处理与日志记录**:良好的异常处理机制能保证程序的稳定运行,而日志记录则有助于排查问题。日志框架如Log4j或SLF4J可以记录系统运行过程中的重要信息。 11. **单元测试与集成测试**:JUnit或TestNG用于...

    Java web项目-高校学生选课系统项目源码

    Java提供了try-catch-finally结构来处理运行时异常,而在Web应用中,我们还需要考虑如何向用户友好地呈现错误信息。 9. **单元测试与集成测试**:为了保证项目的稳定性和质量,开发者会使用JUnit进行单元测试,测试...

    java web 多线程下载器

    综上所述,Java Web多线程下载器是一个综合运用了Java并发编程、文件操作、网络通信等多个核心领域的项目。通过学习和实践此类项目,开发者可以深入理解Java的多线程编程和Web开发技术,提升编程技能。

Global site tag (gtag.js) - Google Analytics