SpringBoot多種自定義錯誤頁面方式小結

在項目中為瞭友好化,對於錯誤頁面,我們常常會使用自定義的頁面。SSM框架組合時代,我們通常通過攔截或者在web.xml中設置對於錯誤碼的錯誤頁面,然而到瞭SpringBoot,web.xml消失瞭,SpringBootServletInitializer初始化servlet代替瞭web.xml。難道要再把web.xml加回去?這樣雖然可以做到,但並不合理。

下面提供瞭多種在SpringBoot中實現自定義錯誤頁面的方法。

以前web.xml方式

先來看下在web.xml中配置錯誤頁面的方式:

<error-page>
    <error-code>404</error-code>
    <location>/error/404.jsp</location>
</error-page>

SpringBoot中實現方式

在SpringBoot後,可以通過如下幾種方式實現自定義錯誤頁面。

1.實現EmbeddedServletContainerCustomizer的bean

適合內嵌服務器,先在controller中定義我們的錯誤頁面Mapping,通過在配置類中實現EmbeddedServletContainerCustomizer的bean,加入對應狀態碼的錯誤頁面。註意這種方式在打成war後,供外部tomcat使用時,將會失效。

定義錯誤頁面:

   @RequestMapping(value = "/error/[code]")
    public String error(@PathVariable int code, Model model) {
        String pager = "/content/error-pager";
        switch (code) {
            case 404:
                model.addAttribute("code", 404);
                pager = "/content/error-pager";
                break;
            case 500:
                model.addAttribute("code", 500);
                pager = "/content/error-pager";
                break;
        }
        return pager;
    }

在配置類中加入EmbeddedServletContainerCustomizer:

  /**
     * 配置默認錯誤頁面(僅用於內嵌tomcat啟動時)
     * 使用這種方式,在打包為war後不起作用
     *
     * @return
     */  
@Bean
public EmbeddedServletContainerCustomizer containerCustomizer() {
        return container -> {
            ErrorPage error404Page = new ErrorPage(HttpStatus.NOT_FOUND, "/error/404");
            ErrorPage error500Page = new ErrorPage(HttpStatus.INTERNAL_SERVER_ERROR, "/error/500");
            container.addErrorPages(error404Page, error500Page);
        };

2.通過攔截器方式

適合內嵌Tomcat或者war方式。

/**
 * @author hgs
 * @version ErrorPageInterceptor.java, v 0.1 2018/03/04 20:52 hgs Exp $
 * <p>
 * 錯誤頁面攔截器
 * 替代EmbeddedServletContainerCustomizer在war中不起作用的方法
 */
@Component
public class ErrorPageInterceptor extends HandlerInterceptorAdapter {
    private List<Integer> errorCodeList = Arrays.asList(404, 403, 500, 501);
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws
        Exception {
       if (errorCodeList.contains(response.getStatus())) {
            response.sendRedirect("/error/" + response.getStatus());
            return false;
        }
        return super.preHandle(request, response, handler);
    }
}

在配置類中添加攔截

@Configuration
public class WebMvcConfig extends WebMvcConfigurerAdapter {
 @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(errorPageInterceptor);//.addPathPatterns("/action/**", "/mine/**");默認所有
        super.addInterceptors(registry);
    }
}

3.自定義靜態error頁面方法

在resource/templates下添加error.html頁面,springBoot會自動找到該頁面作為錯誤頁面,適合內嵌Tomcat或者war方式。

SpringBoot錯誤視圖提供瞭以下錯誤屬性:

  • timestamp:錯誤發生時間;
  • status:HTTP狀態嗎;
  • error:錯誤原因;
  • exception:異常的類名;
  • message:異常消息(如果這個錯誤是由異常引起的);
  • errors:BindingResult異常裡的各種錯誤(如果這個錯誤是由異常引起的);
  • trace:異常跟蹤信息(如果這個錯誤是由異常引起的);
  • path:錯誤發生時請求的URL路徑。

SpringBoot使用的前端框架模板不同,頁面的名稱也有所不同:

  • 實現Spring的View接口的Bean,其ID需要設置為error(由Spring的BeanNameViewResolver所解析);
  • 如果配置瞭Thymeleaf,則需命名為error.html的Thymeleaf模板;
  • 如果配置瞭FreeMarker,則需命名為error.ftl的FreeMarker模板;
  • 如果配置瞭Velocity,則需命名為error.vm的Velocity模板;
  • 如果是用JSP視圖,則需命名為error.jsp的JSP模板。

Thymeleaf實例:

<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head>
    <title th:text="${title}"></title>
</head>
<body class="layout">
<div class="wrap">
    <!-- S top -->
    <div th:include="/header/module-header::module-header"></div>
    <!-- S 內容 -->
    <div class="panel-l container clearfix">
        <div class="error">
            <p class="title"><span class="code" th:text="${status}"></span>非常抱歉,沒有找到您要查看的頁面</p>
            <a href="/" rel="external nofollow"  class="btn-back common-button">返回首頁
                <img class="logo-back" src="/img/back.png">
            </a>
            <div class="common-hint-word">
                <div th:text="${#dates.format(timestamp,'yyyy-MM-dd HH:mm:ss')}"></div>
                <div th:text="${messages}"></div>
                <div th:text="${error}"></div>
            </div>
        </div>
    </div>
</div>
</div>
</body>
</html>

對於外部Tomcat第三中方案是比較推薦的一種實現方式,但不夠靈活,我們不好定義自己的屬性,如果想對其做相應修改,可以參見源碼BasicErrorController,通過繼承AbstractErrorController,並重寫errorHtml方法,達到自己想要的效果。在內嵌Tomcat時,第一種推薦使用,更具靈活性。

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: