ここでは、コントローラがModelAndViewを返し、JSPビューページのレンダリング中に例外がスローされます。理由は、メッセージコードが見つからないという理由があります。
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
<!-- This will catch Http 500 error -->
<error-page>
<error-code>500</error-code>
<location>/WEB-INF/views/jsp/error.jsp</location>
</error-page>
<!-- This will catch any exception -->
<error-page>
<location>/WEB-INF/views/jsp/error.jsp</location>
</error-page>
3.サーブレットフィルタ
また、フィルタを作成して、Webアプリケーション上のリクエスト全体をインターセプトすることもできます。例外がスローされた場合は、ログに記録するか、必要な操作を行います。
package com.mkyong.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.mkyong.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
<!-- must register this, else WebApplicationContext not found -->
<listener>
<listener-class>
org.springframework.web.context.ContextLoaderListener
</listener-class>
</listener>
<filter>
<filter-name>errorHandlerFilter</filter-name>
<filter-class>com.mkyong.form.web.ErrorHandleFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>errorHandlerFilter</filter-name>
<url-pattern>/** </url-pattern>
</filter-mapping>
または、イニシャライザクラスで宣言します。
package com.mkyong.form.config.servlet3;
import javax.servlet.Filter;
import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;
import com.mkyong.form.web.ErrorHandleFilter;
public class MyWebInitializer extends
AbstractAnnotationConfigDispatcherServletInitializer {
//...
@Override
protected Filter[]getServletFilters() {
return new Filter[]{new ErrorHandleFilter()};
}
}
参考文献
Spring MVCでの処理]。 link://spring-mvc/how-to-register-a-servlet-filter-in-spring-mvc/[どのようにして
Spring MVCにサーブレットフィルタを登録する]