Spring MVC 例外を均一に処理する 3 つの方法

(*-*)浩
リリース: 2019-08-29 16:22:01
転載
3437 人が閲覧しました

Spring には、統合例外処理の 3 つのメソッドがあります。つまり、

@ExceptionHandler アノテーションを使用する、HandlerExceptionResolver インターフェイスを実装する、および @controlleradvice アノテーションを使用する

Spring MVC 例外を均一に処理する 3 つの方法

#@ExceptionHandler アノテーションを使用する

このアノテーションを使用する場合の欠点の 1 つは、例外処理のメソッドがエラーの原因となったメソッドと同じコントローラー内に存在する必要があることです。次のように使用します。

@Controller      
public class GlobalController {
/**    
     * 用于处理异常的    
     * @return    
     */      
@ExceptionHandler({MyException.class})
public String exception(MyException e) {
System.out.println(e.getMessage());
e.printStackTrace();
return "exception";
}
@RequestMapping("test")
public void test() {
throw new MyException("出错了!");
}
}
ログイン後にコピー

ご覧のとおり、このメソッドの最大の欠点は、例外をグローバルに制御できないことです。各クラスは 1 回記述する必要があります。

HandlerExceptionResolver インターフェイスを実装する

このメソッドにより、グローバルな例外制御が可能になります。例:

@Component  
public class ExceptionTest implements HandlerExceptionResolver{
/**  
     * TODO 简单描述该方法的实现功能(可选).  
     * @see org.springframework.web.servlet.HandlerExceptionResolver#resolveException(javax.servlet.http.HttpServletRequest, javax.servlet.http.HttpServletResponse, java.lang.Object, java.lang.Exception)  
     */  
public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler,  
            Exception ex) {
System.out.println("This is exception handler method!");
return null;
}
}
ログイン後にコピー

@ControllerAdvice @ExceptionHandler アノテーションを使用します

@ExceptionHandler で説明したように、例外処理を必要とするメソッドは、例外処理を必要とするメソッドと同じコントローラー内に存在する必要があります。エラーが発生しました。コードで @ControllerAdvice を追加する場合、同じコントローラー内にある必要はありません。これも Spring 3.2 によってもたらされた新機能です。名前からわかるように、一般的にはコントローラーの強化を意味します。つまり、@controlleradvice @ExceptionHandler はグローバル例外キャッチを実装することもできます。

この WebExceptionHandle クラスをスキャンして Spring コンテナにロードできることを確認してください。

@ControllerAdvice
@ResponseBody
public class WebExceptionHandle {
private static Logger logger = LoggerFactory.getLogger(WebExceptionHandle.class);
/**
     * 400 - Bad Request
     */
@ResponseStatus(HttpStatus.BAD_REQUEST)
@ExceptionHandler(HttpMessageNotReadableException.class)
public ServiceResponse handleHttpMessageNotReadableException(HttpMessageNotReadableException e) {
logger.error("参数解析失败", e);
return ServiceResponseHandle.failed("could_not_read_json");
}
/**
     * 405 - Method Not Allowed
     */
@ResponseStatus(HttpStatus.METHOD_NOT_ALLOWED)
@ExceptionHandler(HttpRequestMethodNotSupportedException.class)
public ServiceResponse handleHttpRequestMethodNotSupportedException(HttpRequestMethodNotSupportedException e) {
logger.error("不支持当前请求方法", e);
return ServiceResponseHandle.failed("request_method_not_supported");
}
/**
     * 415 - Unsupported Media Type
     */
@ResponseStatus(HttpStatus.UNSUPPORTED_MEDIA_TYPE)
@ExceptionHandler(HttpMediaTypeNotSupportedException.class)
public ServiceResponse handleHttpMediaTypeNotSupportedException(Exception e) {
logger.error("不支持当前媒体类型", e);
return ServiceResponseHandle.failed("content_type_not_supported");
}
/**
     * 500 - Internal Server Error
     */
@ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
@ExceptionHandler(Exception.class)
public ServiceResponse handleException(Exception e) {
if (e instanceof BusinessException){
return ServiceResponseHandle.failed("BUSINESS_ERROR", e.getMessage());
}
logger.error("服务运行异常", e);
e.printStackTrace();
return ServiceResponseHandle.failed("server_error");
}
}
ログイン後にコピー

処理される例外の型が @ExceptionHandler アノテーションで宣言されていない場合、デフォルトでパラメータ リスト内の例外の型が使用されます。したがって、次のように書くこともできます。

@ControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler()
@ResponseBody
String handleException(Exception e){
return "Exception Deal! " + e.getMessage();
}
}
ログイン後にコピー

パラメータ オブジェクトは、コントローラー層によってスローされる例外オブジェクトです。

ResponseEntityExceptionHandler クラスを継承して、Rest インターフェイスのグローバル例外キャプチャを実装し、カスタム形式を返すことができます:

@Slf4j
@ControllerAdvice
public class ExceptionHandlerBean  extends ResponseEntityExceptionHandler {
/**
     * 数据找不到异常
     * @param ex
     * @param request
     * @return
     * @throws IOException
     */
@ExceptionHandler({DataNotFoundException.class})
public ResponseEntity<Object> handleDataNotFoundException(RuntimeException ex, WebRequest request) throws IOException {
return getResponseEntity(ex,request,ReturnStatusCode.DataNotFoundException);
}
/**
     * 根据各种异常构建 ResponseEntity 实体. 服务于以上各种异常
     * @param ex
     * @param request
     * @param specificException
     * @return
     */
private ResponseEntity<Object> getResponseEntity(RuntimeException ex, WebRequest request, ReturnStatusCode specificException) {
ReturnTemplate returnTemplate = new ReturnTemplate();
returnTemplate.setStatusCode(specificException);
returnTemplate.setErrorMsg(ex.getMessage());
return handleExceptionInternal(ex, returnTemplate,
new HttpHeaders(), HttpStatus.OK, request);
}
}
ログイン後にコピー

以上がSpring MVC 例外を均一に処理する 3 つの方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

関連ラベル:
ソース:csdn.net
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート
私たちについて 免責事項 Sitemap
PHP中国語ウェブサイト:福祉オンライン PHP トレーニング,PHP 学習者の迅速な成長を支援します!