在我们写controller层的时候,常常会有这样的困惑,如果需要返回一个数据是,可能为了统一回去构造一个类似下列的数据格式:
{ status:true, msg:"保存成功!", data:[]}
而且在写的时候,可能中间会有很多异常情况处理,需要一直写这样格式的返回数据,能不能统一处理,写controller层的时候,只关心怎么data数据要怎么写,至于其他的自动处理掉呢,spring 3.2的时候提供了一个注解@ControllerAdvice 和 @ExceptionHandler顾名思义就是控制增强和异常处理器,在拦截器层面把返回的报文统一,同时可以对异常情况的报文进行规范。
在网上找到一段一个大神写的,异常处理的例子,首先定义了一个常用的异常状态工具类:
package com.drskj.apiservice.common.utils;import java.lang.reflect.Field;import java.text.SimpleDateFormat;import java.util.ArrayList;import java.util.Date;import java.util.HashMap;import java.util.List;import java.util.Map;import com.alibaba.fastjson.JSON;import com.google.common.collect.Maps;//格式化返回客户端数据格式(json)public class ReturnFormat { private static MapmessageMap = Maps.newHashMap(); //初始化状态码与文字说明 static { messageMap.put("0", ""); messageMap.put("400", "Bad Request!"); messageMap.put("401", "NotAuthorization"); messageMap.put("405", "Method Not Allowed"); messageMap.put("406", "Not Acceptable"); messageMap.put("500", "Internal Server Error"); messageMap.put("1000", "[服务器]运行时异常"); messageMap.put("1001", "[服务器]空值异常"); messageMap.put("1002", "[服务器]数据类型转换异常"); messageMap.put("1003", "[服务器]IO异常"); messageMap.put("1004", "[服务器]未知方法异常"); messageMap.put("1005", "[服务器]数组越界异常"); messageMap.put("1006", "[服务器]网络异常"); } public static String retParam(int status,Object data) { OutputJson json = new OutputJson(status, messageMap.get(String.valueOf(status)), data); return json.toString(); }}
其实我觉得上面用枚举类会不会更好,个人建议啦,然后定义好了常用的异常响应码和提示后,开始写控制器增强类,对应常用的服务器端异常情况的处理:
package com.drskj.apiservice.handler;import java.io.IOException;import org.springframework.beans.ConversionNotSupportedException;import org.springframework.beans.TypeMismatchException;import org.springframework.http.converter.HttpMessageNotReadableException;import org.springframework.http.converter.HttpMessageNotWritableException;import org.springframework.web.HttpMediaTypeNotAcceptableException;import org.springframework.web.HttpRequestMethodNotSupportedException;import org.springframework.web.bind.MissingServletRequestParameterException;import org.springframework.web.bind.annotation.ControllerAdvice;import org.springframework.web.bind.annotation.ExceptionHandler;import org.springframework.web.bind.annotation.ResponseBody;import com.drskj.apiservice.common.utils.ReturnFormat;/** * 异常增强,以JSON的形式返回给客服端 * 异常增强类型:NullPointerException,RunTimeException,ClassCastException, NoSuchMethodException,IOException,IndexOutOfBoundsException 以及springmvc自定义异常等,如下:SpringMVC自定义异常对应的status code Exception HTTP Status Code ConversionNotSupportedException 500 (Internal Server Error)HttpMessageNotWritableException 500 (Internal Server Error)HttpMediaTypeNotSupportedException 415 (Unsupported Media Type)HttpMediaTypeNotAcceptableException 406 (Not Acceptable)HttpRequestMethodNotSupportedException 405 (Method Not Allowed)NoSuchRequestHandlingMethodException 404 (Not Found) TypeMismatchException 400 (Bad Request)HttpMessageNotReadableException 400 (Bad Request)MissingServletRequestParameterException 400 (Bad Request) * */@ControllerAdvicepublic class RestExceptionHandler{ //运行时异常 @ExceptionHandler(RuntimeException.class) @ResponseBody public String runtimeExceptionHandler(RuntimeException runtimeException) { return ReturnFormat.retParam(1000, null); } //空指针异常 @ExceptionHandler(NullPointerException.class) @ResponseBody public String nullPointerExceptionHandler(NullPointerException ex) { ex.printStackTrace(); return ReturnFormat.retParam(1001, null); } //类型转换异常 @ExceptionHandler(ClassCastException.class) @ResponseBody public String classCastExceptionHandler(ClassCastException ex) { ex.printStackTrace(); return ReturnFormat.retParam(1002, null); } //IO异常 @ExceptionHandler(IOException.class) @ResponseBody public String iOExceptionHandler(IOException ex) { ex.printStackTrace(); return ReturnFormat.retParam(1003, null); } //未知方法异常 @ExceptionHandler(NoSuchMethodException.class) @ResponseBody public String noSuchMethodExceptionHandler(NoSuchMethodException ex) { ex.printStackTrace(); return ReturnFormat.retParam(1004, null); } //数组越界异常 @ExceptionHandler(IndexOutOfBoundsException.class) @ResponseBody public String indexOutOfBoundsExceptionHandler(IndexOutOfBoundsException ex) { ex.printStackTrace(); return ReturnFormat.retParam(1005, null); } //400错误 @ExceptionHandler({HttpMessageNotReadableException.class}) @ResponseBody public String requestNotReadable(HttpMessageNotReadableException ex){ System.out.println("400..requestNotReadable"); ex.printStackTrace(); return ReturnFormat.retParam(400, null); } //400错误 @ExceptionHandler({TypeMismatchException.class}) @ResponseBody public String requestTypeMismatch(TypeMismatchException ex){ System.out.println("400..TypeMismatchException"); ex.printStackTrace(); return ReturnFormat.retParam(400, null); } //400错误 @ExceptionHandler({MissingServletRequestParameterException.class}) @ResponseBody public String requestMissingServletRequest(MissingServletRequestParameterException ex){ System.out.println("400..MissingServletRequest"); ex.printStackTrace(); return ReturnFormat.retParam(400, null); } //405错误 @ExceptionHandler({HttpRequestMethodNotSupportedException.class}) @ResponseBody public String request405(){ System.out.println("405..."); return ReturnFormat.retParam(405, null); } //406错误 @ExceptionHandler({HttpMediaTypeNotAcceptableException.class}) @ResponseBody public String request406(){ System.out.println("404..."); return ReturnFormat.retParam(406, null); } //500错误 @ExceptionHandler({ConversionNotSupportedException.class,HttpMessageNotWritableException.class}) @ResponseBody public String server500(RuntimeException runtimeException){ System.out.println("500..."); return ReturnFormat.retParam(406, null); }}
返回格式实体类OutPutJson;这里用到了知名的fastjson将对象转json:
package com.drskj.apiservice.common.utils;import java.io.Serializable;import com.alibaba.fastjson.JSON;public class OutputJson implements Serializable{ /** * 返回客户端统一格式,包括状态码,提示信息,以及业务数据 */ private static final long serialVersionUID = 1L; //状态码 private int status; //必要的提示信息 private String message; //业务数据 private Object data; public OutputJson(int status,String message,Object data){ this.status = status; this.message = message; this.data = data; } public int getStatus() { return status; } public void setStatus(int status) { this.status = status; } public String getMessage() { return message; } public void setMessage(String message) { this.message = message; } public Object getData() { return data; } public void setData(Object data) { this.data = data; } public String toString(){ if(null == this.data){ this.setData(new Object()); } return JSON.toJSONString(this); }}
最后自定义一个Controller层的注解,在实现HandlerExceptionResolver 时候,实现对异常的处理(这个还没有想好怎么实现)。