Spring Boot缺省錯誤視圖解析器
Web應用在處理請求的過程當中發生錯誤是很是常見的狀況,SpringBoot中爲咱們實現了一個錯誤視圖解析器(DefaultErrorViewResolver)。它基於一些常見的約定,嘗試根據HTTP錯誤狀態碼解析出錯誤處理視圖。它會在目錄/error下針對提供的HTTP錯誤狀態碼搜索模板或者靜態資源,好比,給定了HTTP狀態碼404,它會嘗試搜索以下模板或者靜態資源:javascript
若是找不到就用默認的白標錯誤視圖,以下圖所示:html
所以,爲了給用戶最佳的使用體驗,404等常見錯誤須要咱們自定義頁面來處理。如下是幾種自定義錯誤頁面的方式。java
方式1. 定義靜態的錯誤頁面
在 resources 下的 static 目錄下,新建 error 目錄,在其中新建各類靜態錯誤頁面,如 404、500,也能夠模糊處理,如4xx、5xx 等,當程序運行出錯時,會自動根據錯誤代碼(如500)找到相應的錯誤頁面(如/static/error/500.html),給予展現。web
方式2. 定義動態的錯誤頁面(有采用模板引擎)
在有使用模板的狀況下,SpringBoot缺省的錯誤視圖解析器也會在/<templates>/error下搜索錯誤展現視圖。咱們可使用項目中的視圖模板引擎在錯誤頁面來定製展現咱們的錯誤消息。
1) 在 resources 下的 templates 目錄下,新建 error 目錄,在其中新建各類靜態錯誤頁面,如 404、500,也能夠模糊處理,如4xx、5xx 等(與方式1一致)+spring
在模板引擎的支持下能夠取到錯誤的一些信息,並定製化顯示在頁面上,以下(freemarker模板):chrome
錯誤信息定製:app
方式3. 自定義實現錯誤視圖解析,統一錯誤處理
若是不想要使用缺省的錯誤處理視圖解析器,想要定製一些本身的東西(好比說:錯誤引導信息等),按照官方文檔的建議咱們能夠自定義實現錯誤視圖解析接口來處理。
下面就是經過實現錯誤視圖解析接口ErrorViewResolver,將4xx、5xx 的錯誤頁面集中在一個自定義視圖上:
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
錯誤頁面的展現優先級
一、精確大於模糊
二、動態大於靜態