`

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

    博客分类:
  • Java
阅读更多

在实际的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;
			}

 

 

 

以上是个人拙见,勿拍砖,谢谢。

 

  • 大小: 18.4 KB
  • 大小: 42.4 KB
  • 大小: 22.9 KB
  • 大小: 25.2 KB
分享到:
评论
22 楼 jwx0925 2012-12-29  
isaiahzhong 写道
           不明白,假如dao出现了异常,在代码会返回到action调用的地方?
          另外一个问题,出现了错误,你你相当于是跳转到一个错误页面,假如我想alert一个弹出框,然后再弹出框里面显示报错信息,如何实现?


可以封装一下你的exception,使拦截器能够识别你这个exception需要alert一下。
21 楼 isaiahzhong 2012-12-28  
           不明白,假如dao出现了异常,在代码会返回到action调用的地方?
          另外一个问题,出现了错误,你你相当于是跳转到一个错误页面,假如我想alert一个弹出框,然后再弹出框里面显示报错信息,如何实现?
20 楼 hwh0919 2011-10-11  
仁兄,我做项目过程中后台验证出错,用异常的方式抛出,这样会不会影响系统的性能呢?
19 楼 freesky_zh 2011-10-03  
总结得不错。
18 楼 zk7019311 2011-06-13  
学到了不少东西啊
17 楼 mib168 2011-06-13  
一般都是service记录异常网上抛,struts来跳转 不错 回头整理
16 楼 白糖_ 2011-06-11  
itlangqun 写道
楼主可否发个简单示例,我们进行扩充与自定义。


我的这篇文章有源码:
http://www.iteye.com/topic/1069749
15 楼 jwx0925 2011-06-11  
pml346680914 写道
楼主说说简单的jsp+javabean架构如何处理异常??


可以用Servlet的filter吧~封装一个拦截器
14 楼 itlangqun 2011-06-11  
楼主可否发个简单示例,我们进行扩充与自定义。
13 楼 lucky16 2011-06-11  
首先感谢楼主的分享,这个问题我曾经也思考了一下,但是始终没有找到觉得完美的方法!
12 楼 pml346680914 2011-06-10  
楼主说说简单的jsp+javabean架构如何处理异常??
11 楼 melin 2011-06-10  
我现在也是这么处理,包含如果是ajax的处理。我用的是spring mvc

//~ Methods ========================================================================================================
	@ExceptionHandler()
	public @ResponseBody String handle(Exception exception, HttpServletRequest request, HttpServletResponse response) {
		logger.error(request.getRequestURI() + " 请求失败", exception);
		
		ResponseData data = new ResponseData(false, exception.getClass() + ": " + exception.getMessage());
		data.setRequestURI(request.getRequestURI());
		
		StackTraceElement[] trace = exception.getStackTrace();
		StringBuilder traceContent = new StringBuilder();
        for (int i=0; i < trace.length; i++)
        	traceContent.append("\tat " + trace[i]);
		data.setExecptionTrace(traceContent.toString());
		
		String json = "";
		try {
			json = mapper.writeValueAsString(data);
		} catch (Exception e1) {
			e1.printStackTrace();
		}
		
		if(!(request.getRequestURI().endsWith(".json") || request.getRequestURI().endsWith("Json")))
			throw new RuntimeException(exception);
		
		response.setStatus(500);//服务端处理失败
		response.setContentType("application/json;charset=UTF-8");
		return json;
	}


如果是ajax请求,提示处理:
//Ajax请求完成执行。判断Session超时,如果超时或者无效,返回的内容为login.jsp页面的内容,页面包含:AJAX-AccessDeniedException
//如果页面包含AJAX-AccessDeniedException,说明session超时或者无效。
Ext.Ajax.on('requestcomplete', function(conn, response, options){
	if(options.params['REQUEST_MODE'] == "AJAX" &&
		response.responseText.indexOf("AJAX-AccessDeniedException") != -1) {
		Ext.Msg.alert('提示', '会话超时,请重新登录!', function(){
            window.location = './login'; 
        });
	}
});

Ext.Ajax.on('requestexception', function(conn, response, options){
	//ajax请求,出现异常,弹出窗口提示信息。
	//var json = response.responseText.replace("</generated></generated>", "");
	var json = response.responseText;
	var data = Ext.decode(json);

	Ext.Msg.alert('提示', "请求URI:"+ data.requestURI + "<br>错误信息:"+ 
		data.message + "<br><a href='#'>错误堆栈信息</a>", function(){
    });
	return false;
});
10 楼 cobb.chan 2011-06-10  
基本和我使用的一致,另外DAO层是否使用关键还是要看分层情况
9 楼 白糖_ 2011-06-10  
highsense 写道
白糖_ 写道
damoqiongqiu 写道
不错,思路清晰,格式良好
如果要针对不同的异常类型做出不同的响应又该如何呢?


这位朋友过奖了。

不同异常类型依然通过instanceof来判断,判断后就可以做不同操作
if (e instanceof Exception1) {
  Exception1 e1 = (Exception1) e;
  //做其它不不同响应
} else if(e instanceof Exception2){
  Exception2re = (Exception2)e;
  //做其它不不同响应
}

个人认为,根据异常类的类型来区分异常是不是有些臃肿并且难以扩展。我比较喜欢的做法是,自己定义一个BaseException,继承自RuntimeException,然后给BaseException里弄一个errorCode,用errorCode来区分异常类型,这样你只需要维护一张errorCode与异常类型的对应表,就不用定义那么多Exception类了,维护方便,扩展也方便。抛异常的时候只需设置errorCode,捕获异常之后根据errorCode去判断异常类型并做相应处理。


这位朋友跟我想法一致,目前我们公司项目做整体架构的时候,我就是这样设计的。我上面的意思是在开发过程中,我们可能需要引用其它的组件,而其它的组件的自定义异常是不依赖于目前项目的异常的,所以需要分开处理。
就行下图这样,我们整个项目的异常父类是DywlException(继承自RuntimeException),另外我们把系统权限OSPMS组件整合进来了,所以还需要对权限的异常进行处理。



8 楼 highsense 2011-06-10  
白糖_ 写道
damoqiongqiu 写道
不错,思路清晰,格式良好
如果要针对不同的异常类型做出不同的响应又该如何呢?


这位朋友过奖了。

不同异常类型依然通过instanceof来判断,判断后就可以做不同操作
if (e instanceof Exception1) {
  Exception1 e1 = (Exception1) e;
  //做其它不不同响应
} else if(e instanceof Exception2){
  Exception2re = (Exception2)e;
  //做其它不不同响应
}

个人认为,根据异常类的类型来区分异常是不是有些臃肿并且难以扩展。我比较喜欢的做法是,自己定义一个BaseException,继承自RuntimeException,然后给BaseException里弄一个errorCode,用errorCode来区分异常类型,这样你只需要维护一张errorCode与异常类型的对应表,就不用定义那么多Exception类了,维护方便,扩展也方便。抛异常的时候只需设置errorCode,捕获异常之后根据errorCode去判断异常类型并做相应处理。
7 楼 huhuanqadn 2011-06-10  
yangdefeng95802 写道
怎么还用DAO呢,现在多数都不要这个层了!


同样求解
6 楼 newwpp 2011-06-10  
yangdefeng95802 写道
怎么还用DAO呢,现在多数都不要这个层了!

这个说法求解
5 楼 yangdefeng95802 2011-06-09  
怎么还用DAO呢,现在多数都不要这个层了!
4 楼 白糖_ 2011-06-09  
damoqiongqiu 写道
不错,思路清晰,格式良好
如果要针对不同的异常类型做出不同的响应又该如何呢?


这位朋友过奖了。

不同异常类型依然通过instanceof来判断,判断后就可以做不同操作
if (e instanceof Exception1) {
  Exception1 e1 = (Exception1) e;
  //做其它不不同响应
} else if(e instanceof Exception2){
  Exception2re = (Exception2)e;
  //做其它不不同响应
}
3 楼 damoqiongqiu 2011-06-09  
不错,思路清晰,格式良好
如果要针对不同的异常类型做出不同的响应又该如何呢?

相关推荐

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

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

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

     出来工作一年时间了,我也大概对异常处理有了一些了解,在这呢小弟简单介绍下个人对异常处理的见解,抛砖引玉,希望各位大神提出宝贵的意见和建议。  拿spring+struts2+hibernate项目说明:通常一个页面请求到...

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

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

    基于MVC的java Web设计与开发

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

    精通Struts_基于MVC的Java Web设计与开发

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

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

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

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

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

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

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

    精通 Struts:基于 MVC 的 JavaWeb 设计与开发(PDF)

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

    java 面试题 总结

    java编译器要求方法必须声明抛出可能发生的非运行时异常,但是并不要求必须声明抛出未被捕获的运行时异常。 6、说出Servlet的生命周期,并说出Servlet和CGI的区别。 Servlet被服务器实例化后,容器运行其init方法,...

    工程硕士学位论文 基于Android+HTML5的移动Web项目高效开发探究

    1.2.4 Android移动Web项目开发的三种解决方案:Native, Web和Hybrid优缺陷分析 4 1.2.5国内外应用现状 6 1.2.6 研究现状总结 7 1.3研究目标与内容 7 1.3.1多窗口浏览器模式的实现机制 7 1.3.2跨域交互即缓存处理方法...

    超级有影响力霸气的Java面试题大全文档

    java编译器要求方法必须声明抛出可能发生的非运行时异常,但是并不要求必须声明抛出未被捕获的运行时异常。 9、说出Servlet的生命周期,并说出Servlet和CGI的区别。  Servlet被服务器实例化后,容器运行其init方法...

    JavaWeb后台开发框架-其他

    严谨安全: 清晰的系统执行流程,严谨的异常检测和安全机制,详细的日志统计,为系统保驾护航。组件化: 完善的组件化设计,丰富的表单组件,让开发列表和表单更得心应手。无需前端开发,省时省力。简单上手快: ...

    《程序天下:J2EE整合详解与典型案例》光盘源码

    8.4.4 改变异常处理的方式 8.5 小结 第九章 CVS使用指南 9.1 CVS介绍 9.1.1 CVS简介 9.1.2 为什么要使用CVS 9.2 建立CVS的开发环境 9.2.1 下载CVS 9.2.2 配置CVS 9.3 CVS的使用方法 9.3.1 添加CVS资源库 9.3.2 提交...

    搞定J2EE:STRUTS+SPRING+HIBERNATE整合详解与典型案例 (2)

    8.4.4 改变异常处理的方式 8.5 小结 第九章 CVS使用指南 9.1 CVS介绍 9.1.1 CVS简介 9.1.2 为什么要使用CVS 9.2 建立CVS的开发环境 9.2.1 下载CVS 9.2.2 配置CVS 9.3 CVS的使用方法 9.3.1 添加CVS资源库 9.3.2 提交...

    搞定J2EE:STRUTS+SPRING+HIBERNATE整合详解与典型案例 (1)

    8.4.4 改变异常处理的方式 8.5 小结 第九章 CVS使用指南 9.1 CVS介绍 9.1.1 CVS简介 9.1.2 为什么要使用CVS 9.2 建立CVS的开发环境 9.2.1 下载CVS 9.2.2 配置CVS 9.3 CVS的使用方法 9.3.1 添加CVS资源库 9.3.2 提交...

    搞定J2EE:STRUTS+SPRING+HIBERNATE整合详解与典型案例 (3)

    8.4.4 改变异常处理的方式 8.5 小结 第九章 CVS使用指南 9.1 CVS介绍 9.1.1 CVS简介 9.1.2 为什么要使用CVS 9.2 建立CVS的开发环境 9.2.1 下载CVS 9.2.2 配置CVS 9.3 CVS的使用方法 9.3.1 添加CVS资源库 9.3.2 提交...

    Spring.3.x企业应用开发实战(完整版).part2

     Spring3.0引入了众多Java开发者翘首以盼的新功能和新特性,如OXM、校验及格式化框架、REST风格的Web编程模型等。这些新功能实用性强、易用性高,可大幅降低Java应用,特别是JavaWeb应用开发的难度,同时有效提升...

    Spring3.x企业应用开发实战(完整版) part1

     Spring3.0引入了众多Java开发者翘首以盼的新功能和新特性,如OXM、校验及格式化框架、REST风格的Web编程模型等。这些新功能实用性强、易用性高,可大幅降低Java应用,特别是JavaWeb应用开发的难度,同时有效提升...

Global site tag (gtag.js) - Google Analytics