在web项目开发中,一个系统应该要考虑到异常情况的处理,并且应该当异常发生时应该需要记录相应的异常日志,对于用户而言则不能直接抛出异常,需要考虑到用户的体验:
以下就介绍基于spring框架的基础上的异常处理:
1:如何将异常优雅地显示给用户?
如果将一连串的java报错展示给用户,对用户而言是没有意义的,并且也是危险的;
1)定义错误代码:这里以一个枚举类来封装用户错误代码:ErrorCode:
public enum ErrorCode { NULL_OBJ("LUO001","对象为空"), ERROR_ADD_USER("LUO002","添加用户失败"), UNKNOWN_ERROR("LUO999","系统繁忙,请稍后再试...."), FILE_NOT_FOUND("FILE001","你要读取的资源找不到"); private String value; private String desc; private ErrorCode(String value, String desc) { this.setValue(value); this.setDesc(desc); }
自定义项目的业务异常类:继承 RuntimeException
public class BusinessException extends RuntimeException { private static final long serialVersionUID = 1L; public BusinessException(Object Obj) { super(Obj.toString()); } }
自定义异常处理类:实现 HandlerExceptionResolver 这个类是用来处理异常发生时做出的逻辑处理,和spring框架集成:
public class MySimpleMappingExceptionResolver implements HandlerExceptionResolver{ @Override public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object object, Exception exception) { // 判断是否ajax请求 if (!(request.getHeader("accept").indexOf("application/json") > -1 || (request .getHeader("X-Requested-With") != null && request.getHeader( "X-Requested-With").indexOf("XMLHttpRequest") > -1))) { // 如果不是ajax,JSP格式返回 // 为安全起见,只有业务异常我们对前端可见,否则否则统一归为系统异常 Map<String, Object> map = new HashMap<String, Object>(); map.put("success", false); if (exception instanceof BusinessException) { map.put("errorMsg", exception.getMessage()); } else { map.put("errorMsg", "系统异常!"); } //这里需要手动将异常打印出来,由于没有配置log,实际生产环境应该打印到log里面 exception.printStackTrace(); //对于非ajax请求,我们都统一跳转到error.jsp页面 return new ModelAndView("/error", map); } else { // 如果是ajax请求,JSON格式返回 try { response.setContentType("application/json;charset=UTF-8"); PrintWriter writer = response.getWriter(); Map<String, Object> map = new HashMap<String, Object>(); map.put("success", false); // 为安全起见,只有业务异常我们对前端可见,否则统一归为系统异常 if (exception instanceof BusinessException) { map.put("errorMsg", exception.getMessage()); } else { map.put("errorMsg", "系统异常!"); } JSONObject json =new JSONObject(); // writer.write(json.fromObject(map)); // writer.flush(); // writer.close(); } catch (IOException e) { e.printStackTrace(); } } return null; } }
在spring容器中注册这个异常处理类
<!-- 定义异常处理器 -->
<bean id="exceptionResolver" class="bz.beppe.exception.MySimpleMappingExceptionResolver"></bean>
这样,在项目抛出自定义的业务异常时,都或进入这个异常处理类进行处理,然后展示给用户;
当然项目还有一些unchecked Exception,由于代码不要求强制处理,当这些异常发生时,用户就会面对 404 500等服务器内部的页面,目前大多数服务器也都支持在Web.xml中通过<error-page>(Websphere/Weblogic)或者<error-code>(Tomcat)节点配置特定异常情况的显示页面。修改web.xml文件,增加以下内容:
<!-- 出错页面定义 --> <error-page> <exception-type>java.lang.Throwable</exception-type> <location>/500.jsp</location> </error-page> <error-page> <error-code>500</error-code> <location>/500.jsp</location> </error-page> <error-page> <error-code>404</error-code> <location>/404.jsp</location> </error-page>
这样一个健全的异常处理系统就好了;
以下是controller层对这些业务异常和系统异常 以及 unchecked exception 的测试代码:
public class ExceptionController { @RequestMapping(value = "/controller", method = RequestMethod.GET) public void controller(Model model,String id) { throw new BusinessException(ErrorCode.NULL_OBJ.toString()); } @RequestMapping(value = "/service", method = RequestMethod.GET) public void service(Model model,String id) { // String str=null; // str.substring(10); // System.out.println("代码运行了"); File file=new File("c:/a.txt"); try { FileReader reader = new FileReader(file); int i = reader.read(); System.out.println(i); } catch (FileNotFoundException e) { // TODO Auto-generated catch block System.out.println("文件找不到"); throw new BusinessException(ErrorCode.FILE_NOT_FOUND); // e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block System.out.println("读取文件出错"); e.printStackTrace(); } } }
至此,一个简单的完整的异常处理系统就完成了;