成人视屏在线观看-国产99精品-国产精品1区2区-欧美一级在线观看-国产一区二区日韩-色九九九

您的位置:首頁技術(shù)文章
文章詳情頁

Springboot錯誤處理機(jī)制實現(xiàn)原理解析

瀏覽:6日期:2023-05-24 13:13:22

1.默認(rèn)的錯誤機(jī)制

默認(rèn)效果

①在瀏覽器中訪問不存在的請求時,springboot默認(rèn)返回一個空白頁面

Springboot錯誤處理機(jī)制實現(xiàn)原理解析

瀏覽器的請求頭

Springboot錯誤處理機(jī)制實現(xiàn)原理解析

②客戶端訪問時,返回json數(shù)據(jù)

{ 'timestamp': '2020-03-24T02:49:56.572+0000', 'status': 404, 'error': 'Not Found', 'message': 'No message available', 'path': '/'}

客戶端訪問的請求頭

Springboot錯誤處理機(jī)制實現(xiàn)原理解析

原理

可以參照 ErrorMvcAutoConfiguration 錯誤處理的自動配置

給容器中添加了以下組件

1.DefaultErrorAttributes

public Map<String, Object> getErrorAttributes(WebRequest webRequest, boolean includeStackTrace) { Map<String, Object> errorAttributes = new LinkedHashMap(); errorAttributes.put('timestamp', new Date()); this.addStatus(errorAttributes, webRequest); this.addErrorDetails(errorAttributes, webRequest, includeStackTrace); this.addPath(errorAttributes, webRequest); return errorAttributes;}

@RequestMapping( produces = {'text/html'} ) public ModelAndView errorHtml(HttpServletRequest request, HttpServletResponse response) { HttpStatus status = this.getStatus(request); //處理頁面的請求返回給前臺數(shù)據(jù) model 的獲取 ,調(diào)用 Map<String, Object> model = Collections.unmodifiableMap(this.getErrorAttributes(request, this.isIncludeStackTrace(request, MediaType.TEXT_HTML))); response.setStatus(status.value()); ModelAndView modelAndView = this.resolveErrorView(request, response, status, model); return modelAndView != null ? modelAndView : new ModelAndView('error', model); } //調(diào)用 AbstractErrorController#getErrorAttributes protected Map<String, Object> getErrorAttributes(HttpServletRequest request, boolean includeStackTrace) { WebRequest webRequest = new ServletWebRequest(request); return this.errorAttributes.getErrorAttributes(webRequest, includeStackTrace); } 最終調(diào)用DefaultErrorAttributes#getErrorAttributes public Map<String, Object> getErrorAttributes(WebRequest webRequest, boolean includeStackTrace) {

2.BasicErrorController : 處理默認(rèn)的 /error 請求

@Controller@RequestMapping({'${server.error.path:${error.path:/error}}'}) public class BasicErrorController extends AbstractErrorController { private final ErrorProperties errorProperties;public String getErrorPath() { return this.errorProperties.getPath();}@RequestMapping( produces = {'text/html'} //產(chǎn)生html類型的數(shù)據(jù),瀏覽器發(fā)送的請求來到這個方法處理)public ModelAndView errorHtml(HttpServletRequest request, HttpServletResponse response) {//獲取狀態(tài)碼 HttpStatus status = this.getStatus(request);//獲取模型數(shù)據(jù) Map<String, Object> model = Collections.unmodifiableMap(this.getErrorAttributes(request, this.isIncludeStackTrace(request, MediaType.TEXT_HTML))); response.setStatus(status.value());//去哪個頁面作為錯誤頁面,包括頁面地址和內(nèi)容 ModelAndView modelAndView = this.resolveErrorView(request, response, status, model); return modelAndView != null ? modelAndView : new ModelAndView('error', model);}@RequestMapping //產(chǎn)生json類型的數(shù)據(jù), 其他客戶端發(fā)送的請求來到這個方法處理public ResponseEntity<Map<String, Object>> error(HttpServletRequest request) { HttpStatus status = this.getStatus(request); if (status == HttpStatus.NO_CONTENT) { return new ResponseEntity(status); } else { Map<String, Object> body = this.getErrorAttributes(request, this.isIncludeStackTrace(request, MediaType.ALL)); return new ResponseEntity(body, status); }}

3.ErrorPageCustomizer

public class ErrorProperties { @Value('${error.path:/error}') private String path = '/error'; //系統(tǒng)出現(xiàn)錯誤請求之后來到 /error 請求進(jìn)行處理 ,(類似于以前 web.xml 中注冊的錯誤頁面規(guī)則)

4.DefaultErrorViewResolver

public ModelAndView resolveErrorView(HttpServletRequest request, HttpStatus status, Map<String, Object> model) { ModelAndView modelAndView = this.resolve(String.valueOf(status.value()), model); if (modelAndView == null && SERIES_VIEWS.containsKey(status.series())) { modelAndView = this.resolve((String)SERIES_VIEWS.get(status.series()), model); } return modelAndView; } private ModelAndView resolve(String viewName, Map<String, Object> model) {//默認(rèn) springboot 可以找到這個頁面 error/404 String errorViewName = 'error/' + viewName;//模板引擎可以解析這個頁面地址就用模板引擎解析 TemplateAvailabilityProvider provider = this.templateAvailabilityProviders.getProvider(errorViewName, this.applicationContext);//模板引擎可用的情況下就返回到 errorViewName 指定的視圖地址 return provider != null ? new ModelAndView(errorViewName, model) : this.resolveResource(errorViewName, model); }  //模板引擎不可用就在靜態(tài)資源文件夾里面找 errorViewName 對應(yīng)的頁面 error/404.html private ModelAndView resolveResource(String viewName, Map<String, Object> model) { String[] var3 = this.resourceProperties.getStaticLocations(); int var4 = var3.length; for(int var5 = 0; var5 < var4; ++var5) { String location = var3[var5]; try {Resource resource = this.applicationContext.getResource(location);resource = resource.createRelative(viewName + '.html');//如果靜態(tài)資源文件中由 這個資源就直接使用,否則返回為空if (resource.exists()) { return new ModelAndView(new DefaultErrorViewResolver.HtmlResourceView(resource), model);} } catch (Exception var8) { } }  return null; }

步驟:

一旦系統(tǒng)出現(xiàn) 4xx 或者 5xx 之類的錯誤,ErrorPageCustomizer 就會生效(定制錯誤的響應(yīng)規(guī)則),就會來到 /error 請求,會被BasicErrorController

處理。

①響應(yīng)頁面 去哪個頁面由 DefaultErrorViewResolver 決定

protected ModelAndView resolveErrorView(HttpServletRequest request, HttpServletResponse response, HttpStatus status, Map<String, Object> model) { Iterator var5 = this.errorViewResolvers.iterator(); //解析所有的 ErrorViewResolver 得到 modelAndView ModelAndView modelAndView; do { if (!var5.hasNext()) { return null; } ErrorViewResolver resolver = (ErrorViewResolver)var5.next(); modelAndView = resolver.resolveErrorView(request, status, model); } while(modelAndView == null); return modelAndView;}

2.錯誤信息的定制

①如何定制錯誤頁面

1>有模板引擎的情況下: error/狀態(tài)碼 ;【將錯誤頁面命名為 錯誤碼.html 放在模板引擎文件夾下的 error 文件夾下】,發(fā)生此狀態(tài)碼的錯誤就來到

對應(yīng)的頁面;

我們可以使用 4xx 和 5xx 作為錯誤頁面的文件名來匹配這種類型的所欲錯誤,精確優(yōu)先(優(yōu)先尋找精確的 狀態(tài)碼.html );

頁面能夠獲取到的信息

timestamp :時間戳

status : 狀態(tài)碼

exception : 異常對象

message : 異常消息

errors : JSR303數(shù)據(jù)校驗的錯誤都在這兒

2>.沒有模板引擎(模板引擎找不到這個頁面),靜態(tài)資源文件夾下找

3>.以上都沒有錯誤頁面,就默認(rèn)來到 springboot 默認(rèn)的錯誤頁面

②、自定義異常處理&返回定制json數(shù)據(jù);

@ControllerAdvicepublic class MyExceptionHandler { @ResponseBody @ExceptionHandler(UserNotExistException.class) public Map<String,Object> handleException(Exception e){ Map<String,Object> map = new HashMap<>(); map.put('code','user.notexist'); map.put('message',e.getMessage()); return map; }}//通過異常處理器,但沒有自適應(yīng)效果(瀏覽器返回頁面,客戶端訪問返回json數(shù)據(jù))

2)、轉(zhuǎn)發(fā)到/error進(jìn)行自適應(yīng)響應(yīng)效果處理

@RequestMapping( produces = {'text/html'})public ModelAndView errorHtml(HttpServletRequest request, HttpServletResponse response) { //獲取錯誤的狀態(tài)碼,在分析的過程中,要注意參數(shù)從哪兒來? =======》前領(lǐng)導(dǎo)的一句話,哈哈…… HttpStatus status = this.getStatus(request); Map<String, Object> model = Collections.unmodifiableMap(this.getErrorAttributes(request, this.isIncludeStackTrace(request, MediaType.TEXT_HTML))); response.setStatus(status.value());//依據(jù)錯誤狀態(tài)碼解析錯誤試圖,如果直接轉(zhuǎn)發(fā),不指定錯誤狀態(tài)碼則試圖解析出錯(直接轉(zhuǎn)發(fā)狀態(tài)碼為 200 ,到不了定制的 4xx 5xx 的頁面) ModelAndView modelAndView = this.resolveErrorView(request, response, status, model); return modelAndView != null ? modelAndView : new ModelAndView('error', model);}

@ExceptionHandler(UserNotExistException.class) public String handleException(Exception e, HttpServletRequest request){ Map<String,Object> map = new HashMap<>(); <strong>//傳入我們自己的錯誤狀態(tài)碼 4xx 5xx,否則就不會進(jìn)入定制錯誤頁面的解析流程</strong> /** * Integer statusCode = (Integer) request .getAttribute('javax.servlet.error.status_code'); */ request.setAttribute('javax.servlet.error.status_code',500); map.put('code','user.notexist'); map.put('message',e.getMessage()); //轉(zhuǎn)發(fā)到/error return 'forward:/error'; }

3)、將我們的定制數(shù)據(jù)攜帶出去;======》即修改model中的值即可

出現(xiàn)錯誤以后,會來到/error請求,會被BasicErrorController處理,響應(yīng)出去可以獲取的數(shù)據(jù)是由getErrorAttributes得到的(是AbstractErrorController(ErrorController)規(guī)定的方法);

1、完全來編寫一個ErrorController的實現(xiàn)類【或者是編寫AbstractErrorController的子類】,放在容器中;

2、頁面上能用的數(shù)據(jù),或者是json返回能用的數(shù)據(jù)都是通過errorAttributes.getErrorAttributes得到;

容器中DefaultErrorAttributes.getErrorAttributes();默認(rèn)進(jìn)行數(shù)據(jù)處理的;

自定義ErrorAttributes

//給容器中加入我們自己定義的ErrorAttributes@Componentpublic class MyErrorAttributes extends DefaultErrorAttributes { @Override public Map<String, Object> getErrorAttributes(RequestAttributes requestAttributes, boolean includeStackTrace) { Map<String, Object> map = super.getErrorAttributes(requestAttributes, includeStackTrace); map.put('company','atguigu'); return map; }}

最終的效果:響應(yīng)是自適應(yīng)的,可以通過定制ErrorAttributes改變需要返回的內(nèi)容,

Springboot錯誤處理機(jī)制實現(xiàn)原理解析

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。

標(biāo)簽: Spring
相關(guān)文章:
主站蜘蛛池模板: 一区二区三区在线播放 | 97免费公开视频 | 免费视频一区二区 | 日韩三级一区二区 | 国产最新网站 | 中文字幕天堂最新版在线网 | 亚洲视频精品 | 欧美在线高清视频播放免费 | 美女黄网站色一级毛片 | 国产精品夫妇久久 | 一区二区三区视频在线 | 污全彩肉肉无遮挡彩色 | 精品一区二区三区波多野结衣 | 亚洲男人天堂久久 | 中文一区在线 | 69交性视频| 久久中文亚洲国产 | 免费一级毛片在线播放 | 亚洲欧美一级久久精品 | 中国一级特黄视频 | a级精品九九九大片免费看 a级毛片免费观看网站 | 午夜影院亚洲 | 国内国语一级毛片在线视频 | 亚洲一区免费在线 | 久草中文视频 | 又黄又免费的网站 | 最近中文字幕在线 | 中文 | 国产精品久久毛片蜜月 | 高清不卡毛片免费观看 | 成年网站视频在线观看 | 国产91精品久久久久久久 | 欧美做爰xxxⅹ性欧 欧美做爰免费大片在线观看 | 日本一级毛片高清免费观看视频 | 精品欧美高清一区二区免费 | 亚洲性爰视频 | 欧美一级级毛片 | 一二三中文乱码亚洲乱码 | 免费观看成人www精品视频在线 | 国产成人精品本亚洲 | 欧产日产国产精品精品 | 日韩字幕一中文在线综合 |