SpringBoot2零基礎(chǔ)到精通之異常處理與web原生組件注入
1 異常處理
??默認(rèn)情況下,SpringBoot會提供/error處理所有的錯誤請求并返回相應(yīng)的信息,對于瀏覽器客戶端來說會返回一個包含時間戳、狀態(tài)碼、錯誤信息、攜帶的自定義異常信息、發(fā)生錯誤的路徑等信息的錯誤Whitelabel頁面,對于機(jī)器客戶端(postman等)會返回一個包含以上內(nèi)容的JSON數(shù)據(jù)
1.1 異常處理之錯誤頁面
??要想替代之前瀏覽器客戶端返回的錯誤Whitelabel頁面,需要將自定義的html頁面放在靜態(tài)資源static等的error文件夾下或者模板引擎templates的error文件夾下,這樣的話出現(xiàn)錯誤時SpringBoot發(fā)送/error請求就會自動解析這些頁面進(jìn)行渲染。頁面解析規(guī)則:先將狀態(tài)碼的值與error文件夾下的頁面名進(jìn)行精確匹配,如果精確匹配不到的話就按照4xx、5xx這樣的方式進(jìn)行模糊匹配,要是還匹配不到的話就返回Whitelabel頁面

1.2 異常處理之精確捕獲
??要是說錯誤頁面是按照狀態(tài)碼進(jìn)行頁面處理的話,精確捕獲就是通過異常類進(jìn)行捕獲,捕獲之后再進(jìn)行一系列的自定義操作。具體步驟就是:創(chuàng)建一個異常處理器類,并在類上加@ControllerAdvice注解表明是一個異常處理器并向容器中注冊該組件,@ExceptionHandler注解對參數(shù)中的異常類進(jìn)行精準(zhǔn)捕獲,并在方法體定義具體的處理操作。
@Slf4j
@ControllerAdvice
public class GlobalExceptionHandler {
@ExceptionHandler({ArithmeticException.class, NullPointerException.class})
public String handleArithException(Exception e) {
log.info("系統(tǒng)捕獲到異常信息:{}", e);
return "login";
}
}
1.3 異常處理之自定義異常
??有時候我們需要在程序中自定義一些運(yùn)行時異常,這些異常并不會像那些異常一樣產(chǎn)生異常狀態(tài)碼,甚至在未定義之前都不算是異常且不會影響程序的正常運(yùn)行。這時就需要我們自定義異常的產(chǎn)生邏輯,并自定義異常類創(chuàng)建有參無參構(gòu)造器,在類上加@ResponseStatus注解,使用注解參數(shù)定義異常響應(yīng)碼和異常信息
@ResponseStatus(value = HttpStatus.FORBIDDEN, reason = "用戶數(shù)量太多")
public class UserTooManyException extends RuntimeException{
public UserTooManyException() {
}
public UserTooManyException(String message) {
super(message);
}
}
異常產(chǎn)生邏輯:
// 判斷用戶數(shù)量拋出用戶數(shù)量過多的自定義異常
if (users.size() > 3) {
throw new UserTooManyException();
}
1.4 異常處理之框架底層異常
??除了exception類中定義的異常外,spring框架底層也定義了一些異常,這些異常由DefaultHandlerExceptionResolver來處理
2 web原生組件的注入
2.1 servlet組件
??servlet組件需要自定義創(chuàng)建一個servlet類繼承HttpServlet,并使用@WebServlet注解的urlPatterns屬性聲明攔截的請求,再通過主程序類上使用@ServletComponentScan(basePackages = “…”)注解將該組件掃描注冊到容器中。
// 聲明攔截的請求
@WebServlet(urlPatterns = "/my")
public class MyServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.getWriter().write("385695");
}
}
2.2 filter組件
??filter組件需要自定義創(chuàng)建一個filter類實(shí)現(xiàn)Filter接口,并使用@WebFilter注解的urlPatterns屬性聲明過濾的請求
@Slf4j
@WebFilter(urlPatterns = {"/css/*", "/images/*"})
public class MyFilter implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
log.info("MyFilter初始化……");
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
log.info("MyFilter方法開始工作了……");
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {
log.info("MyFilter銷毀了……");
}
}
2.3 listener組件
??listener組件需要自定義創(chuàng)建一個listener類實(shí)現(xiàn)ServletContextListener 接口,并使用@WebListener注解
@Slf4j
@WebListener
public class MyServletContextListener implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent sce) {
log.info("MyServletContextListener監(jiān)聽到項(xiàng)目初始化完成……");
}
@Override
public void contextDestroyed(ServletContextEvent sce) {
log.info("MyServletContextListener監(jiān)聽到項(xiàng)目已經(jīng)銷毀……");
}
}
除了使用注解進(jìn)行注冊之外,還可以使用配置類的方式將以上三種組件注冊到容器中去
@Configuration
public class MyRegistConfig {
@Bean
public ServletRegistrationBean MyServlet() {
MyServlet myServlet = new MyServlet();
return new ServletRegistrationBean(myServlet, "/my", "/my02");
}
@Bean
public FilterRegistrationBean myFilter() {
MyFilter myFilter = new MyFilter();
FilterRegistrationBean filterRegistrationBean = new FilterRegistrationBean(myFilter);
filterRegistrationBean.setUrlPatterns(Arrays.asList("/my", "/my02"));
return filterRegistrationBean;
}
@Bean
public ServletListenerRegistrationBean myListener() {
MyServletContextListener listener = new MyServletContextListener();
return new ServletListenerRegistrationBean(listener);
}
}
??通過servlet組件聲明的/my請求并不會經(jīng)過spring的攔截器攔截處理,而是直接交由tomcat服務(wù)器進(jìn)行處理:現(xiàn)在有這么兩個組件攔截到我們發(fā)送的/my請求,一個是spring的組件DispatcherServlet通過/路徑攔截到,一個是tomcat的自定義MyServlet組件通過/my路徑攔截到。tomcat服務(wù)器對請求有這么一個處理規(guī)則,當(dāng)多個servlet組件都能處理到同一個請求的時候,使用匹配度最高的組件進(jìn)行處理,也就是說處理請求的是tomcat的MyServlet組件。
3 web實(shí)現(xiàn)定制化總結(jié)
- 創(chuàng)建配置類實(shí)現(xiàn)WebMvcConfigurer接口,重寫相應(yīng)的方法或者使用@Bean注解向容器中擴(kuò)展功能(使用的最多)
- 自定義配置類使用@Bean注解使用自定義的web原生組件替換容器中的默認(rèn)組件或者向容器中添加組件
- 配置文件的配置項(xiàng)修改
- xxxCustomizer@EnableWebMvc注解表示全面接管SpringMVC,導(dǎo)致的效果就是SpringBoot的所有自動配置全部失效,全部功能都需要自己進(jìn)行配置。大佬專屬注解,小白的禁術(shù)!?。?/li>
到此這篇關(guān)于SpringBoot2零基礎(chǔ)到精通之異常處理與web原生組件注入的文章就介紹到這了,更多相關(guān)SpringBoot2 異常處理內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring?Boot?詳細(xì)分析Conditional自動化配置注解
首先我們先了解一下@Conditional注解,@Conditional是Spring4新提供的注解,它的作用是按照一定的條件進(jìn)行判斷,需要注入的Bean滿足給定條件才可以注入到Spring?IOC容器中2022-07-07
java?實(shí)現(xiàn)獲取指定位置后的第一個數(shù)字
這篇文章主要介紹了java?實(shí)現(xiàn)獲取指定位置后的第一個數(shù)字,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-01-01
java使用XSSFWorkbook實(shí)現(xiàn)讀寫Excel
這篇文章主要為大家詳細(xì)介紹了java如何通過使用XSSFWorkbook實(shí)現(xiàn)讀寫Excel功能,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-04-04
SpringBoot @ConfigurationProperties使用詳解
這篇文章主要介紹了SpringBoot @ConfigurationProperties使用詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02

