Spring Boot の API インターフェイスの非侵襲的実装は、統一された JSON 形式で応答を返します。

はじめに: プロジェクトにおいて、API インターフェースが一貫性のない応答を返すと、フロントエンドとバックエンドが分離されているシーンで不可解なバグが発生し、すべてのインターフェースを変更すると作業負荷が小さくないため、非侵入的な方法を採用します。解決策は、統一された JSON 形式で応答を返す API インターフェイスを実装することです。

戻り値の JSON ボディを定義する

{
    
      
    "code": 200,  // 状态码
    "message": "success",  // 返回信息描述
    "data": {
    
    }  // 返回数据
}  

JavaBeanフィールドを定義する

@Getter  
@ToString  
public class Result<T> {
    
      
    /** 业务错误码 */  
    private Integer code;  
    /** 信息描述 */  
    private String message;  
    /** 返回参数 */  
    private T data;  
  
    private Result(ResultStatus resultStatus, T data) {
    
      
        this.code = resultStatus.getCode();  
        this.message = resultStatus.getMessage();  
        this.data = data;  
    }  
  
    /** 业务成功返回业务代码和描述信息 */  
    public static Result<Void> success() {
    
      
        return new Result<Void>(ResultStatus.SUCCESS, null);  
    }  
  
    /** 业务成功返回业务代码,描述和返回的参数 */  
    public static <T> Result<T> success(T data) {
    
      
        return new Result<T>(ResultStatus.SUCCESS, data);  
    }  
  
    /** 业务成功返回业务代码,描述和返回的参数 */  
    public static <T> Result<T> success(ResultStatus resultStatus, T data) {
    
      
        if (resultStatus == null) {
    
      
            return success(data);  
        }  
        return new Result<T>(resultStatus, data);  
    }  
  
    /** 业务异常返回业务代码和描述信息 */  
    public static <T> Result<T> failure() {
    
      
        return new Result<T>(ResultStatus.INTERNAL_SERVER_ERROR, null);  
    }  
  
    /** 业务异常返回业务代码,描述和返回的参数 */  
    public static <T> Result<T> failure(ResultStatus resultStatus) {
    
      
        return failure(resultStatus, null);  
    }  
  
    /** 业务异常返回业务代码,描述和返回的参数 */  
    public static <T> Result<T> failure(ResultStatus resultStatus, T data) {
    
      
        if (resultStatus == null) {
    
      
            return new Result<T>(ResultStatus.INTERNAL_SERVER_ERROR, null);  
        }  
        return new Result<T>(resultStatus, data);  
    }  
}

ここでは統一 JSON 形式を実装しただけですが、問題も見つかりました。統一 JSON 形式を返したい場合は、Result を返す必要があります。当然 Object を返すこともできます。なぜ同じ作業を繰り返す必要があるのでしょうか。解決策はありますか? もちろん解決策はあります。コードの最適化を始めましょう。

最適化され完成された

@ResponseBody アノテーションを使用すると、返されたオブジェクトが JSON 文字列にシリアル化されることは誰もが知っています。まずはこれから始めましょう。これは、シリアル化の前にオブジェクトを結果に割り当てることです。org.springframework.web.servlet. mvc.method を確認できます。 annotation.ResponseBodyAdvice および org.springframework.web.bind.annotation.ResponseBody。

@ResponseBody 継承クラス

@ResponseBody アノテーションから始めて、 @ResponseBody を継承するアノテーション クラスを作成することにしました。これをクラスやメソッドにマークして自由に使用できるようにします。

@Retention(RetentionPolicy.RUNTIME)  
@Target({
    
    ElementType.TYPE, ElementType.METHOD})  
@Documented  
@ResponseBody  
public @interface ResponseResultBody {
    
      
  
}  

ResponseBodyAdvice 継承クラス

@RestControllerAdvice  
public class ResponseResultBodyAdvice implements ResponseBodyAdvice<Object> {
    
      
  
    private static final Class<? extends Annotation> ANNOTATION_TYPE = ResponseResultBody.class;  
  
    /**  
     * 判断类或者方法是否使用了 @ResponseResultBody  
     */  
    @Override  
    public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {
    
      
        return AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ANNOTATION_TYPE) || returnType.hasMethodAnnotation(ANNOTATION_TYPE);  
    }  
  
    /**  
     * 当类或者方法使用了 @ResponseResultBody 就会调用这个方法  
     */  
    @Override  
    public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
    
      
        // 防止重复包裹的问题出现  
        if (body instanceof Result) {
    
      
            return body;  
        }  
        return Result.success(body);  
    }  
}  

返却ごとにResultオブジェクトを返すのではなく、Objectを直接返却することでJSON形式を統一できるのはすごいですね。SpringMVCに直接統合管理を手伝ってもらうのが最適です。

インターフェースだけ見ると、直接例外をスローするインターフェースは helloError と helloMyError ですが、例外リターンの処理が統一されていないようです。

統合された戻り JSON 形式の高度な - 例外処理 (@ExceptionHandler))

例外処理 @ResponseStatus (非推奨)
@ResponseStatus の使用法は次のとおりで、Controller クラス、Controller メソッド、Exception クラスで使用できますが、それでも負荷が非常に大きくなります。

@RestController  
@RequestMapping("/error")  
@ResponseStatus(value = HttpStatus.INTERNAL_SERVER_ERROR, reason = "Java的异常")  
public class HelloExceptionController {
    
      
  
    private static final HashMap<String, Object> INFO;  
  
    static {
    
      
        INFO = new HashMap<String, Object>();  
        INFO.put("name", "galaxy");  
        INFO.put("age", "70");  
    }  
  
    @GetMapping()  
    public HashMap<String, Object> helloError() throws Exception {
    
      
        throw new Exception("helloError");  
    }  
  
    @GetMapping("helloJavaError")  
    @ResponseStatus(value = HttpStatus.INTERNAL_SERVER_ERROR, reason = "Java的异常")  
    public HashMap<String, Object> helloJavaError() throws Exception {
    
      
        throw new Exception("helloError");  
    }  
  
    @GetMapping("helloMyError")  
    public HashMap<String, Object> helloMyError() throws Exception {
    
      
        throw new MyException();  
    }  
}  
  
@ResponseStatus(value = HttpStatus.INTERNAL_SERVER_ERROR, reason = "自己定义的异常")  
class MyException extends Exception {
    
      
  
}  

グローバル例外処理 @ExceptionHandler (推奨)
ResponseResultBodyAdvice クラスを変更するには、コードが少し多すぎます

@Slf4j  
@RestControllerAdvice  
public class ResponseResultBodyAdvice implements ResponseBodyAdvice<Object> {
    
      
  
    private static final Class<? extends Annotation> ANNOTATION_TYPE = ResponseResultBody.class;  
  
    /** 判断类或者方法是否使用了 @ResponseResultBody */  
    @Override  
    public boolean supports(MethodParameter returnType, Class<? extends HttpMessageConverter<?>> converterType) {
    
      
        return AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ANNOTATION_TYPE) || returnType.hasMethodAnnotation(ANNOTATION_TYPE);  
    }  
  
    /** 当类或者方法使用了 @ResponseResultBody 就会调用这个方法 */  
    @Override  
    public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class<? extends HttpMessageConverter<?>> selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {
    
      
        if (body instanceof Result) {
    
      
            return body;  
        }  
        return Result.success(body);  
    }  
  
  
    /**  
     * 提供对标准Spring MVC异常的处理  
     *  
     * @param ex      the target exception  
     * @param request the current request  
     */  
    @ExceptionHandler(Exception.class)  
    public final ResponseEntity<Result<?>> exceptionHandler(Exception ex, WebRequest request) {
    
      
        log.error("ExceptionHandler: {}", ex.getMessage());  
        HttpHeaders headers = new HttpHeaders();  
        if (ex instanceof ResultException) {
    
      
            return this.handleResultException((ResultException) ex, headers, request);  
        }  
        // TODO: 2019/10/05 galaxy 这里可以自定义其他的异常拦截  
        return this.handleException(ex, headers, request);  
    }  
  
    /** 对ResultException类返回返回结果的处理 */  
    protected ResponseEntity<Result<?>> handleResultException(ResultException ex, HttpHeaders headers, WebRequest request) {
    
      
        Result<?> body = Result.failure(ex.getResultStatus());  
        HttpStatus status = ex.getResultStatus().getHttpStatus();  
        return this.handleExceptionInternal(ex, body, headers, status, request);  
    }  
  
    /** 异常类的统一处理 */  
    protected ResponseEntity<Result<?>> handleException(Exception ex, HttpHeaders headers, WebRequest request) {
    
      
        Result<?> body = Result.failure();  
        HttpStatus status = HttpStatus.INTERNAL_SERVER_ERROR;  
        return this.handleExceptionInternal(ex, body, headers, status, request);  
    }  
  
    /**  
     * org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExceptionHandler#handleExceptionInternal(java.lang.Exception, java.lang.Object, org.springframework.http.HttpHeaders, org.springframework.http.HttpStatus, org.springframework.web.context.request.WebRequest)  
     * <p>  
     * A single place to customize the response body of all exception types.  
     * <p>The default implementation sets the {@link WebUtils#ERROR_EXCEPTION_ATTRIBUTE}  
     * request attribute and creates a {@link ResponseEntity} from the given  
     * body, headers, and status.  
     */  
    protected ResponseEntity<Result<?>> handleExceptionInternal(  
            Exception ex, Result<?> body, HttpHeaders headers, HttpStatus status, WebRequest request) {
    
      
  
        if (HttpStatus.INTERNAL_SERVER_ERROR.equals(status)) {
    
      
            request.setAttribute(WebUtils.ERROR_EXCEPTION_ATTRIBUTE, ex, WebRequest.SCOPE_REQUEST);  
        }  
        return new ResponseEntity<>(body, headers, status);  
    }  
}

おすすめ

転載: blog.csdn.net/weixin_44011409/article/details/111829712