异常映射
异常机制是Java程序中针对有可能发生的问题所提前作出的应急解决方案。在SpringMVC中可以通过异常映射的方式,将异常类型和某个视图名称对应起来,让用户不是看到异常信息,而是一个比较友好的界面。
局限性:同步请求需要一个新的页面时这样操作是没问题的,但是对于需要数据片段的异步请求来说,就会导致Ajax请求收到的响应无法解析。
解决方案:
在spring-mvc.xml
<!-- 配置异常映射机制 --> <!-- 为了能够兼容异步请求的异常信息响应格式,所以使用自定义CrowdfundingExceptionResolver类 --> <bean id="simpleMappingExceptionResolver" class="com.crowdfunding.component.resolver.CrowdfundingExceptionResolver"> <property name="exceptionMappings"> <props> <prop key="java.lang.Exception">error</prop> </props> </property> </bean>
分布式架构中没有Spring-mvc.xml那我们应该怎么办呢?
可以使用注解,例如:
@ControllerAdvice public class MyExceptionResolver{ @ExceptionHandler(value=Exception.class) public String resolveException(Exception e, Model model){ model.addAttribute("exception",e); return "error"; } }
重写异常解析器的doResolveException()方法
为什么要重写?
重写示例
public class CrowdfundingExceptionResolver extends SimpleMappingExceptionResolver { @Override protected ModelAndView doResolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { //判断当前请求是否为异步请求.如果 X-Requested-With 为 null,则为同步请求。X-Requested-With 为 XMLHttpRequest 则为 Ajax 请求。 if ((request.getHeader("accept").contains("application/json") || (request.getHeader("X-Requested-With") != null && request.getHeader("X-Requested-With").contains("XMLHttpRequest")))) { try { //创建ResultVO对象用来封装Ajax响应结果 ResultVO<String> resultVO = new ResultVO<>(); resultVO.setResult(ResultVO.FAILED); resultVO.setData(ResultVO.NO_DATA); resultVO.setMessage(ex.getMessage()); //将resultVO对象转换为JSON数据 Gson gson = new Gson(); String json = gson.toJson(resultVO); //设置响应消息头 response.setContentType("application/json;charset=UTF-8"); //将resultVO的JSON数据返回给浏览器 response.getWriter().write(json); } catch (IOException e) { e.printStackTrace(); } return null; } //如果不满足if条件说明是同步请求,则调用父类的方法按照常规方式处理 return super.doResolveException(request, response, handler, ex); } }
Gson的依赖
pom.xml
<dependency> <groupId>com.google.code.gson</groupId> <artifactId>gson</artifactId> </dependency>