Spring Boot自定義錯誤視圖

Spring Boot缺省錯誤視圖解析器
  Web應用在處理請求的過程當中發生錯誤是很是常見的狀況,SpringBoot中爲咱們實現了一個錯誤視圖解析器(DefaultErrorViewResolver)。它基於一些常見的約定,嘗試根據HTTP錯誤狀態碼解析出錯誤處理視圖。它會在目錄/error下針對提供的HTTP錯誤狀態碼搜索模板或者靜態資源,好比,給定了HTTP狀態碼404,它會嘗試搜索以下模板或者靜態資源:javascript

  •  /<templates>/error/404.<ext> - 這裏<templates>表示所配置的模板所在目錄,<ext>表示所用的模板的文件名
  •  /<static>/error/404.html  - 這裏<static>表示靜態資源文件所在路徑、
  • /<templates>/error/4xx.<ext>
  • /<static>/error/4xx.html

若是找不到就用默認的白標錯誤視圖,以下圖所示:html

  

所以,爲了給用戶最佳的使用體驗,404等常見錯誤須要咱們自定義頁面來處理。如下是幾種自定義錯誤頁面的方式。java

 

方式1. 定義靜態的錯誤頁面
resources 下的 static 目錄下,新建 error 目錄,在其中新建各類靜態錯誤頁面,如 404500,也能夠模糊處理,如4xx5xx 等,當程序運行出錯時,會自動根據錯誤代碼(如500)找到相應的錯誤頁面(如/static/error/500.html),給予展現。web

  


方式2. 定義動態的錯誤頁面(有采用模板引擎)
在有使用模板的狀況下,SpringBoot缺省的錯誤視圖解析器也會在/<templates>/error下搜索錯誤展現視圖。咱們可使用項目中的視圖模板引擎在錯誤頁面來定製展現咱們的錯誤消息。
1) 在 resources 下的 templates 目錄下,新建 error 目錄,在其中新建各類靜態錯誤頁面,如 404500,也能夠模糊處理,如4xx5xx 等(與方式1一致)+spring

  

在模板引擎的支持下能夠取到錯誤的一些信息,並定製化顯示在頁面上,以下(freemarker模板):chrome

  

  錯誤信息定製:app

  • timestamp:時間戳
  • status:狀態碼
  • error:錯誤提示
  • exception:異常對象
  • trace:跟蹤流程日誌,404狀態下無
  • message:異常消息
  • path:請求路徑

 

方式3. 自定義實現錯誤視圖解析,統一錯誤處理
  若是不想要使用缺省的錯誤處理視圖解析器,想要定製一些本身的東西(好比說:錯誤引導信息等),按照官方文檔的建議咱們能夠自定義實現錯誤視圖解析接口來處理。
下面就是經過實現錯誤視圖解析接口ErrorViewResolver,將4xx5xx 的錯誤頁面集中在一個自定義視圖上:
1)實現 ErrorViewResolver 接口ide

package com.hongyang.admin.web;

import org.springframework.boot.autoconfigure.web.servlet.error.ErrorViewResolver;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.ModelAndView;

import javax.servlet.http.HttpServletRequest;
import java.util.Map;

/**
 * 實現自定義的錯誤視圖解析器
 */
@Component
public class AdminErrorViewResolver implements ErrorViewResolver {
    /**
     * 實現ErrorViewResolver約定方法,
     * 返回統一的錯誤視圖.
     * @param request
     * @param status
     * @param model
     * @return
     */
    @Override
    public ModelAndView resolveErrorView(HttpServletRequest request, HttpStatus status, Map<String, Object> model) {
        return new ModelAndView("/error/index", model);
    }
}

 

2)完成錯誤視圖,在templates/error下添加index.ftlh視圖(freemarker模板)ui

<!DOCTYPE html>

<html>
<head >
    <link href="/content/public/images/logo-small.png" rel="shortcut icon" />
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    <title>${status}</title>
    <meta name="renderer" content="webkit">
    <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
    <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
    <meta name="apple-mobile-web-app-status-bar-style" content="black">
    <meta name="apple-mobile-web-app-capable" content="yes">
    <meta name="format-detection" content="telephone=no">
    <style>
        body {
            position: fixed;
            z-index: 10;
            top: 0;
            left: 0;
            right: 0;
            bottom: 0;
            width: 100%;
            padding: 0;
            margin: 0px;
            font-size: 14px;
            background: #fff;
            word-wrap: break-word;
        }

        p {
            padding: 0 15px;
        }

        .btn {
            border: 0px;
            color: #fff;
            cursor: pointer;
            text-align: center;
            background-color: #ff7a5f\0;
            box-shadow: #cccccc 0 2px 15px 0;
            -webkit-box-shadow: 0 2px 7px 0 rgba(0,0,0,0.2);
            background: radial-gradient(circle at 300% 50%, rgb(255, 195, 114) 50%, rgb(255, 105, 90) 100%);
            transition: all .2s ease-out,box-shadow .2s ease-out;
        }

        .btn:hover {
            color: #FFFFFF;
            transform: scale(1.1);
        }

        .btn:focus {
            outline: none;
        }

        .container {
            margin: 8% auto;
        }

        .container p {
            margin: 35px auto;
            text-align: center;
        }

        .container img {
            width: 20%;
        }

        .container .font {
            color: #848484;
        }

        .container .btn-back {
            width: 180px;
            height: 35px;
            border-radius: 6px;
        }

        #container-info {
            display: none;
            position: fixed;
            z-index: 11;
            top: 5%;
            left: 0;
            right: 0;
            margin: 0 auto;
            width: 65%;
            height: 85%;
            overflow: hidden;
            border-radius: 4px;
            border: 1px solid #f1986e;
            background-color: #fff;
            box-shadow: inset 0 1px 1px rgba(0,0,0,.075), 0 0 8px rgb(242, 154, 110);
        }

        #container-info .btn-close {
            position: absolute;
            right: 5px;
            top: 5px;
            width: 25px;
            height: 25px;
            line-height: 25px;
            font-size: 22px;
            padding: 2px;
            border-radius: 50%;
            text-align: center;
        }

        #container-info p {
            font-size: 12px;
            line-height: 20px;
        }

        .show {
            display: block !important;
        }

        .cor-r {
            color: red;
        }

        @media (max-width: 767px) {
            .container img {
                width: 50%;
            }
            #container-info {
                width: 85%;
            }
        }

        .panel-heading {
            height: 32px;
            line-height: 32px;
            padding: 5px 15px;
        }
        .panel-body {
            height: calc(85vh - 40px);
            overflow: auto;
        }
        .panel-orange .panel-heading {
            background-color: #ffa0681f;
            color: #ff6f5c;
        }
    </style>
</head>
<body>
<form id="form1" >
    <div class="container">
        <p><img src="/content/public/images/error_${status}.png"/></p>
        <p class="font">${error},<a onclick="errorDetail(true)" href="javascript: void(0)">點擊查看明細</a>!</p>
        <p><button type="button" class="btn btn-back" id="btnBack" >返回</button></p>
    </div>
    <div id="container-info">
        <span class="btn btn-close" onclick="errorDetail(false)">×</span>
        <div class="panel panel-orange">
            <div class="panel-heading">
                錯誤說明
            </div>
            <div class="panel-body">
                <#if path??>
                    <p><b>請求的URL:</b>${path}</p>
                </#if>
                <#if message??>
                    <p><b>異常信息:</b><span class="cor-r">${message}</span></p>
                </#if>
                <#if trace??>
                    <p><b>StackTrace:</b><br>${trace}</p>
                </#if>
            </div>
        </div>
    </div>
    <script type="text/javascript">
        window.onload = function () {
            var btn = document.getElementById("btnBack");
            btn.onclick = function () {
                var url = document.referrer;
                if (url.indexOf("home/main") > 0) {
                    window.parent.tabDelete();
                    return;
                }
                window.history.back(-1);
            }
        }
        function errorDetail(isShow) {
            var con = document.getElementById("container-info");
            con.className = isShow ? "show" : ""; // 兼容IE8
        }
    </script>
</form>
</body>
</html>

 

3)最後效果url

   

錯誤頁面的展現優先級
一、精確大於模糊
二、動態大於靜態

相關文章
相關標籤/搜索