spring3.0开发不可避免要遇到异常处理,如果只有jsp请求能捕获到异常,是不足够的, 有时候AJAX返回json数据时遇到异常。这时候默认的处理方式不能满足了, 需要自定义的方式支持同步和ajax异步异常处理。
或许大家都知道spring3的异常处理方法有:
1 以配置文件的方式进行异常管理
在配置dispacher-servlet.xml
- <bean id="exceptionResolver" class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
- <property name="defaultErrorView">
- <value>jsonView</value>
- </property>
- <property name="exceptionMappings">
- <props>
- <prop key="java.lang.Exception">/common/error.jsp</prop>
- </props>
- </property>
- </bean>
配置多个异常处理的地方
2 以注解的方式配置异常管理
为了方便阅读, 把代码的地方发全一点
- @Controller
- public class ExceptionHandlerController
- {
- private final Logger log = LoggerFactory.getLogger(getClass());
- @ExceptionHandler(ManagerSecurityException.class)
- public ModelAndView handleManagerSecurityException(ManagerSecurityException e,
- HttpServletResponse response)
- throws IOException
- {
- log.info("Manager exception handler " + e.getMessage());
- response.sendError(HttpServletResponse.SC_FORBIDDEN,
- e.getMessage());
- return new ModelAndView("viewName",new HashMap());
- }
- }
这样的配置处理jstl请求jsp没有问题,
如果是jquery的 $.post的方法请求, 那么如果还是返回 error.jsp , 那在 js的success函数里则不能很好地输出显示了。
问题又来了: 如果在 handleManagerSecurityException方法的上面打上注解 @ResponeBody
那么这样又只能返回 json格式串的异常处理, 你想同步请求跳到到error.jsp又不行啦!
解决办法是这样的:定制化异常处理
- <bean id="exceptionResolver" class="com.wsd.core.web.servlet.handle.CustomSimpleMappingExceptionResolver">
- <property name="defaultErrorView">
- <value>/common/error.jsp</value>
- </property>
- <property name="exceptionMappings">
- <props>
- <prop key="java.lang.Exception">/common/error.jsp</prop>
- </props>
- </property>
- </bean>
下面我们来看看CustomSimpleMappingExceptionResolver.java是如何做到支持JSP和JSON格式返回的异常错误的
- @Override
- protected ModelAndView doResolveException(HttpServletRequest request,
- HttpServletResponse response,
- Object handler,
- Exception ex) {
- // Expose ModelAndView for chosen error view.
- String viewName = determineViewName(ex, request);
- if (viewName != null) {//JSP格式返回
- if(!(request.getHeader("accept").indexOf("application/json")>-1 || request.getHeader("X-Requested-With").indexOf("XMLHttpRequest")>-1)){//如果不是异步请求
- // Apply HTTP status code for error views, if specified.
- // Only apply it if we're processing a top-level request.
- Integer statusCode = determineStatusCode(request, viewName);
- if (statusCode != null) {
- applyStatusCodeIfPossible(request, response, statusCode);
- return getModelAndView(viewName, ex, request);
- }
- }else{//JSON格式返回
- Map<String, Object> model=new HashMap<String, Object>();
- if(this.logger.isDebugEnabled()){
- model.put("debug", true);
- }//exception
- model.put(ConfigConstants.RESULT, ex.getMessage());
- model.put("failure", true);
- try {
- response.getWriter().write("有异常啦!");
- } catch (IOException e) {
- e.printStackTrace();
- }
- return new ModelAndView();
- }
- return null;
- }
- else {
- return null;
- }
- }
这里判断了request.getHeader("accept").indexOf("application/json")是不是异步请求
下面给出同步和异步请求的Header
同步的text/html
=== MimeHeaders ===
host = localhost:8888
user-agent = Mozilla/5.0 (Windows; U; Windows NT 6.1; en-GB; rv:1.9.2.8) Gecko/20100722 Firefox/3.6.8
accept = text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
accept-language = en,zh;q=0.7,en-gb;q=0.3
accept-encoding = gzip,deflate
accept-charset = ISO-8859-1,utf-8;q=0.7,*;q=0.7
keep-alive = 115
connection = keep-alive
cookie = JSESSIONID=BB7441E4B481FF64A5BCC8E6F596C330
异步的application/json
=== MimeHeaders ===
host = localhost:8888
user-agent = Mozilla/5.0 (Windows; U; Windows NT 6.1; en-GB; rv:1.9.2.8) Gecko/20100722 Firefox/3.6.8
accept = application/json, text/javascript, */*
accept-language = en,zh;q=0.7,en-gb;q=0.3
accept-encoding = gzip,deflate
accept-charset = ISO-8859-1,utf-8;q=0.7,*;q=0.7
keep-alive = 115
connection = keep-alive
content-type = application/x-www-form-urlencoded
x-requested-with = XMLHttpRequest
referer = http://localhost:8888/auth/auth/dictionary/dictionaryForm.html
cookie = JSESSIONID=A4B59EA87C9B83B71C0D455634746411
请注意
try {
response.getWriter().write("有异常啦!");
} catch (IOException e) {
e.printStackTrace();
}
return new ModelAndView();
关键就在最后这1句 return new ModelAndView();
为什么不写 return null; 或者 return new ModelAndView('jsonView', model); 呢?
因为spring3的源码里这样写着了
- protected ModelAndView processHandlerException(HttpServletRequest request,
- HttpServletResponse response,
- Object handler,
- Exception ex) throws Exception {
- // Check registerer HandlerExceptionResolvers...
- ModelAndView exMv = null;
- for (HandlerExceptionResolver handlerExceptionResolver : this.handlerExceptionResolvers) {
- exMv = handlerExceptionResolver.resolveException(request, response, handler, ex);
- if (exMv != null) {
- break;
- }
- }
- if (exMv != null) {
- if (exMv.isEmpty()) {
- return null;
- }
- if (logger.isDebugEnabled()) {
- logger.debug("Handler execution resulted in exception - forwarding to resolved error view: " + exMv,
- ex);
- }
- WebUtils.exposeErrorRequestAttributes(request, ex, getServletName());
- return exMv;
- }
- throw ex;
- }
注意看
if (exMv != null) {
if (exMv.isEmpty()) {
return null;
}
这句和最后一句: throw ex;
如果你写 respone.getWriter().write("有异常啦"); 你也要 new一个空的ModelAndView,
上面的代码表明, 如果你 return null; 它就跑出异常 throw ex; 如果没人接收, 则会调用你配置的默认异常处理器
或者中断不做任何输出。
好了, 这样可以同时支持同步和AJAX异步请求异常处理了。
下面附上截图, 以免误人子弟~
相关推荐
2. 高级特性实践:探索Spring Security、Spring WebSocket、Spring Boot等进阶主题,提升项目开发能力。 通过以上学习,你可以全面掌握Spring 2.0的核心技术和应用,为成为熟练的Spring开发者奠定坚实的基础。在...
- **学习笔记_7[1].4__Spring_MVC提供的更多功能**: 这部分可能会介绍Spring MVC的高级特性,比如视图解析器、拦截器、异常处理以及自定义转换器和格式化器。 - **Spring_MVC_3[1].0.5+Spring_3.0.5+MyBatis3.0.4全...
6. **异步处理**:Spring MVC支持Servlet 3.0的异步处理,通过`@Async`注解实现后台任务的异步执行。 接下来,我们讨论文件上传功能: 7. **文件上传API**:在Spring MVC中,`MultipartFile`接口用于处理上传的...
Spring3.0是Spring在积蓄了3年之久后,隆重推出的一个重大升级版本,进一步加强了Spring作为Java领域第一开源平台的翘楚地位。 Spring3.0引入了众多Java开发者翘首以盼的新功能和新特性,如OXM、校验及格式化框架...
在"25号05:Spring MVC1课堂内容"中,我们可以推测这是一个教学资源,涵盖了Spring MVC的基础到进阶知识,可能包括了从版本v01到v02的更新内容,特别是针对测试类的优化和使用。 1. **Spring MVC 概述**: Spring ...
Spring3.0是Spring在积蓄了3年之久后,隆重推出的一个重大升级版本,进一步加强了Spring作为Java领域第一开源平台的翘楚地位。 Spring3.0引入了众多Java开发者翘首以盼的新功能和新特性,如OXM、校验及格式化框架...
这份文档详细涵盖了Java编程语言的各个方面,包括基础概念、进阶特性、并发处理、内存管理、垃圾收集、集合框架、IO流、网络编程、设计模式以及框架应用等核心知识点。以下是对这些关键领域的详细阐述: 1. **Java...
再者,异常处理是Java程序设计中的关键环节。面试中可能会询问如何正确地使用try-catch-finally语句,以及何时使用checked异常和unchecked异常。理解并能灵活运用finally块,以及自定义异常的创建,都是展示你编程...
- 使用`CommonsMultipartResolver`的替代品:Spring Boot 2.1及以上版本推荐使用`MultipartResolver`接口的实现类`StandardServletMultipartResolver`,它集成了Servlet 3.0的多部分解析。 - 大文件上传:对于大文件...
Spring Framework 6.0和Spring Boot 3.0进一步增强了微服务架构的支持。HTTP/2和HTTP/3的使用可以提高网络通信效率。了解如何分析和解决CPU利用率飙升、慢SQL、内存溢出和线程死锁等问题,是系统维护的重要技能。...
- **异常处理**:掌握异常处理机制,提高程序的健壮性。 #### 七、XML技术 ##### 1. XML基础知识 - **文档类型定义(DTD)与模式(Schema)**:理解DTD与Schema的作用及区别。 - **XSLT与XPath**:学习如何使用XSLT...
这份资料可能会涵盖诸如JVM(Java虚拟机)工作原理,内存模型,垃圾回收机制,多线程同步,异常处理,设计模式,数据结构与算法,网络编程,IO与NIO,以及Spring框架等相关内容。面试突击版通常会强调实际应用和解决...
2. **异常处理:** 掌握如何使用try-catch-finally语句,理解Checked和Unchecked异常,以及如何自定义异常。 3. **内存管理:** 理解Java内存模型,包括堆内存、栈内存、方法区、本地方法栈,以及垃圾收集机制和...
JavaSE(Java Standard Edition)是学习Java的第一步,它包括Java语法、面向对象编程概念、异常处理、集合框架、多线程、输入/输出流等基础知识。在学习JavaSE的同时,建议同步学习数据结构与算法,这是提升编程能力...
2. **异常处理**:如何通过try-catch-finally语句块来捕获和处理运行时异常,提高代码的健壮性。 3. **集合框架**:详述ArrayList、LinkedList、HashMap、HashSet等常用集合类的使用,以及集合的遍历、查找、排序...
理解ActionForm、Action和ActionForward的使用,以及如何实现国际化(I18N)和异常处理。 7. **XML**:XML是数据交换和配置文件的标准,学习XML的基础,包括XSL、XSLT、DTD和Schema,虽然Java编程可以直接跳过,但...
9. **chapter15.zip** - 最后一个章节可能涉及高级主题,如Servlet 3.0及以上版本的新特性,如注解驱动的Servlet配置,以及MVC框架如Spring MVC的简要介绍。 10. **JSDGSource.zip** - 这个文件名暗示可能是《Java ...
- 学习异常处理机制,理解如何编写健壮的代码。 - **数据库交互**: - **第四阶段**:学习使用JDBC(Java Database Connectivity)进行数据库操作。JDBC提供了标准接口来访问关系型数据库,包括连接数据库、执行...