Spring MVC –ビューページによってスローされた例外をキャッチする
シナリオは次のとおりです。コントローラーはModelAndViewを返し、JSPビューページのレンダリング中に例外がスローされます。理由は、メッセージコードの1つが見つからないためです。
org.apache.jasper.JasperException: org.springframework.context.NoSuchMessageException: No message found under code 'Diff.userform.password' for locale 'en_US'. org.apache.jasper.servlet.JspServletWrapper.handleJspException(JspServletWrapper.java:549) org.apache.jasper.servlet.JspServletWrapper.service(JspServletWrapper.java:470) org.apache.jasper.servlet.JspServlet.serviceJspFile(JspServlet.java:390) org.apache.jasper.servlet.JspServlet.service(JspServlet.java:334) javax.servlet.http.HttpServlet.service(HttpServlet.java:722)
例外がスローされ、HTTP 500エラーページが直接表示されます。
1. 問題
グローバルに例外ハンドラーを宣言しましたが、それでもNoSuchMessageException
をキャッチできませんか?
@ControllerAdvice public class GlobalExceptionHandler { private final Logger logger = LoggerFactory.getLogger(GlobalExceptionHandler.class); @ExceptionHandler(value = Exception.class) public ModelAndView defaultErrorHandler(HttpServletRequest req, Exception e) throws Exception { logger.error("[URL] : {}", req.getRequestURL(), e); ModelAndView mav = new ModelAndView(); mav.addObject("exception", e); mav.addObject("url", req.getRequestURL()); mav.setViewName("error""); return mav; } }
2. 溶液
いいえ、ビューページによってスローされる例外をキャッチできません。 @ExceptionHandler
は、Controllerクラスによってスローされた例外をキャッチするためにのみ使用されます。
あなたができる最善のことは、web.xml
でエラーページを定義することです
web.xml
500 /WEB-INF/views/jsp/error.jsp /WEB-INF/views/jsp/error.jsp
Note
より良い解決策があるかどうか教えてください:)
3. サーブレットフィルター
または、フィルタを作成して、Webアプリケーションのリクエスト全体をインターセプトできます。 例外がスローされた場合は、ログに記録するか、必要な操作を行います。
package com.example.form.web; import java.io.IOException; import javax.servlet.Filter; import javax.servlet.FilterChain; import javax.servlet.FilterConfig; import javax.servlet.ServletException; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.web.context.support.WebApplicationContextUtils; import com.example.form.service.UserService; public class ErrorHandleFilter implements Filter { private final Logger logger = LoggerFactory.getLogger(ErrorHandleFilter.class); //Your custom bean to handle error ErrorHandlerSpecialForce errHandle; @Override public void destroy() { // ... } @Override public void init(FilterConfig filterConfig) throws ServletException { //Get bean from Spring container errHandle = (ErrorHandlerSpecialForce) WebApplicationContextUtils .getRequiredWebApplicationContext(filterConfig.getServletContext()) .getBean("errorHandlerSpecialForce"); } @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { try { // your code, do whatever errHandle.handle(request, response); chain.doFilter(request, response); } catch (Exception ex) { logger.error("Error : {}", ex); //log it request.setAttribute("errorMessage", ex); request.getRequestDispatcher("/WEB-INF/views/jsp/error.jsp") .forward(request, response); } } }
上記のフィルターを登録するには、web.xml
で宣言します
web.xml
org.springframework.web.context.ContextLoaderListener errorHandlerFilter com.example.form.web.ErrorHandleFilter errorHandlerFilter /*
または、初期化クラスで宣言します。
package com.example.form.config.servlet3; import javax.servlet.Filter; import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer; import com.example.form.web.ErrorHandleFilter; public class MyWebInitializer extends AbstractAnnotationConfigDispatcherServletInitializer { //... @Override protected Filter[] getServletFilters() { return new Filter[]{new ErrorHandleFilter()}; } }