国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

Springboot默認的錯誤頁是如何工作及工作原理你肯定不知道?

開發 前端
到此你就知道了一個錯誤的頁是如何在Springboot中被注冊的。到目前為止我們看到的注冊到tomcat容器中的錯誤頁都是個地址,比如:默認是/error。那這個默認的/error又是怎么提供的接口呢?

環境:Springboot2.4.12

環境配置

接下來的演示都是基于如下接口進行。

@RestController
@RequestMapping("/exceptions")
public class ExceptionsController {
    
  @GetMapping("/index")
  public Object index(int a) {
    if (a == 0) {
      throw new BusinessException() ;
    }
    return "exception" ;
  }
    
}

默認錯誤輸出

默認情況下,當請求一個接口發生異常時會有如下兩種情況的錯誤信息提示

  • 基于HTML

圖片圖片

  • 基于JSON

圖片圖片

上面兩個示例通過請求的Accept請求頭設置希望接受的數據類型,得到不同的響應數據類型。

標準web錯誤頁配置

在標準的java web項目中我們一般是在web.xml文件中進行錯誤頁的配置,如下:

<error-page>
  <location>/error</location>
</error-page>

如上配置后,如發生了異常以后容器會自動地跳轉到錯誤頁面。

Spring實現原理

在Springboot中沒有web.xml,并且Servlet API也沒有提供相應的API進行錯誤頁的配置。那么在Springboot中又是如何實現錯誤頁的配置呢?

Springboot內置了應用服務,如Tomcat,Undertow,Jetty,默認是Tomcat。那接下來看下基于默認的Tomcat容器錯誤頁是如何進行配置的。

  • Servlet Web服務自動配置
@EnableConfigurationProperties(ServerProperties.class)
@Import({ServletWebServerFactoryAutoConfiguration.BeanPostProcessorsRegistrar.class, 
         ServletWebServerFactoryConfiguration.EmbeddedTomcat.class,...})
public class ServletWebServerFactoryAutoConfiguration {
  @Configuration(proxyBeanMethods = false)
  @ConditionalOnClass({ Servlet.class, Tomcat.class, UpgradeProtocol.class })
  @ConditionalOnMissingBean(value = ServletWebServerFactory.class, search = SearchStrategy.CURRENT)
  static class EmbeddedTomcat {


    // 這里主要就是配置Web 容器服務,如這里使用的Tomcat
    // 注意該類實現了ErrorPageRegistry ,那么也就是說該類可以用來注冊錯誤頁的
    @Bean
    TomcatServletWebServerFactory tomcatServletWebServerFactory(
      ObjectProvider<TomcatConnectorCustomizer> connectorCustomizers,
      ObjectProvider<TomcatContextCustomizer> contextCustomizers,
      ObjectProvider<TomcatProtocolHandlerCustomizer<?>> protocolHandlerCustomizers) {
      TomcatServletWebServerFactory factory = new TomcatServletWebServerFactory();
      factory.getTomcatConnectorCustomizers().addAll(connectorCustomizers.orderedStream().collect(Collectors.toList()));
      factory.getTomcatContextCustomizers().addAll(contextCustomizers.orderedStream().collect(Collectors.toList()));
      factory.getTomcatProtocolHandlerCustomizers().addAll(protocolHandlerCustomizers.orderedStream().collect(Collectors.toList()));
      return factory;
    }


  }
}

在@Import中只列出了兩個比較重要的BeanPostProcessorsRegistrar與EmbeddedTomcat

BeanPostProcessorsRegistrar注冊了兩個BeanPostProcessor處理器

public static class BeanPostProcessorsRegistrar implements ImportBeanDefinitionRegistrar, BeanFactoryAware {
  public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
    if (this.beanFactory == null) {
      return;
    }
    registerSyntheticBeanIfMissing(registry, "webServerFactoryCustomizerBeanPostProcessor", WebServerFactoryCustomizerBeanPostProcessor.class, WebServerFactoryCustomizerBeanPostProcessor::new);
    registerSyntheticBeanIfMissing(registry, "errorPageRegistrarBeanPostProcessor", ErrorPageRegistrarBeanPostProcessor.class, ErrorPageRegistrarBeanPostProcessor::new);
  }
}

通過名稱也能知道WebServerFactoryCustomizerBeanPostProcessor用來處理Tomcat相關的自定義信息;ErrorPageRegistrarBeanPostProcessor 這個就是重點了,這個就是用來配置我們的自定義錯誤頁面的。

public class ErrorPageRegistrarBeanPostProcessor implements BeanPostProcessor, BeanFactoryAware {
  @Override
  public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
    // 這里判斷了當前的bean對象是否是ErrorPageRegistry的實例
    // 當前類既然是BeanPostProcessor實例,同時上面注冊了一個TomcatServletWebServerFactory Bean實例
    // 那么在實例化TomcatServletWebServerFactory時一定是會調用該BeanPostProcessor處理器的
    if (bean instanceof ErrorPageRegistry) {
      postProcessBeforeInitialization((ErrorPageRegistry) bean);
    }
    return bean;
  }
  // 注冊錯誤頁面
  private void postProcessBeforeInitialization(ErrorPageRegistry registry) {
    for (ErrorPageRegistrar registrar : getRegistrars()) {
      registrar.registerErrorPages(registry);
    }
  }
  private Collection<ErrorPageRegistrar> getRegistrars() {
    if (this.registrars == null) {
      // Look up does not include the parent context
      // 從當前上下文中(比包括父上下文)查找ErrorPageRegistrar Bean對象
      this.registrars = new ArrayList<>(this.beanFactory.getBeansOfType(ErrorPageRegistrar.class, false, false).values());
      this.registrars.sort(AnnotationAwareOrderComparator.INSTANCE);
      this.registrars = Collections.unmodifiableList(this.registrars);
    }
    return this.registrars;
  }
}

注冊錯誤頁面

在上一步中知道了錯誤頁的注冊入口是在一個ErrorPageRegistrarBeanPostProcessor Bean后處理器中進行注冊的,接下來繼續深入查看這個錯誤頁是如何被注冊的。

接著上一步在ErrorPageRegistrarBeanPostProcessor中查找ErrorPageRegistrar類型的Bean對象。在另外一個自動配置中(ErrorMvcAutoConfiguration)有注冊ErrorPageRegistrar Bean對象

@AutoConfigureBefore(WebMvcAutoConfiguration.class)
@EnableConfigurationProperties({ ServerProperties.class, WebMvcProperties.class })
public class ErrorMvcAutoConfiguration {
  
  // 該類是ErrorPageRegistrar子類,那么在注冊錯誤頁的時候注冊的就是該類中生成的錯誤頁信息
  static class ErrorPageCustomizer implements ErrorPageRegistrar, Ordered {
    private final ServerProperties properties;
    private final DispatcherServletPath dispatcherServletPath;
    protected ErrorPageCustomizer(ServerProperties properties, DispatcherServletPath dispatcherServletPath) {
      this.properties = properties;
      this.dispatcherServletPath = dispatcherServletPath;
    }
    @Override
    public void registerErrorPages(ErrorPageRegistry errorPageRegistry) {
      // 錯誤頁的地址可以在配置文件中自定義server.error.path進行配置,默認:/error
      ErrorPage errorPage = new ErrorPage(this.dispatcherServletPath.getRelativePath(this.properties.getError().getPath()));
      errorPageRegistry.addErrorPages(errorPage);
    }
    @Override
    public int getOrder() {
      return 0;
    }
  }


}

關鍵代碼

//  errorPageRegistry對象的實例是TomcatServletWebServerFactory 
errorPageRegistry.addErrorPages(errorPage);

TomcatServletWebServerFactory中注冊錯誤頁信息,該類的父類(AbstractConfigurableWebServerFactory)方法中有添加錯誤也的方法

public abstract class AbstractConfigurableWebServerFactory {
  private Set<ErrorPage> errorPages = new LinkedHashSet<>();
  public void addErrorPages(ErrorPage... errorPages) {
    this.errorPages.addAll(Arrays.asList(errorPages));
  }
}

這個錯誤頁的注冊到Tomcat容器中又是如何實現的呢?

Tomcat中注冊錯誤頁

接下來看看這個錯誤頁是如何與Tomcat關聯在一起的。

Spring容器最核心的方法是refresh方法

public abstract class AbstractApplicationContext {
  public void refresh() {
    // ...
    // Initialize other special beans in specific context subclasses.
    onRefresh();
    // ...
  }
}

執行onRefresh方法

public class ServletWebServerApplicationContext extends GenericWebApplicationContext {
  protected void onRefresh() {
    super.onRefresh();
    try {
      // 創建Tomcat服務
      createWebServer();
    } catch (Throwable ex) {
      throw new ApplicationContextException("Unable to start web server", ex);
    }
  }
  private void createWebServer() {
    // ...
    // 返回應用于創建嵌入的Web服務器的ServletWebServerFactory。默認情況下,此方法在上下文本身中搜索合適的bean。
    // 在上面ServletWebServerFactoryAutoConfiguration自動配置中,已經自動的根據當前的環境創建了TomcatServletWebServerFactory對象
    ServletWebServerFactory factory = getWebServerFactory();
    // 獲取WebServer實例, factory = TomcatServletWebServerFactory
    this.webServer = factory.getWebServer(getSelfInitializer());
    // ...
  }
}

調用TomcatServletWebServerFactory#getWebServer方法

public class TomcatServletWebServerFactory extends AbstractServletWebServerFactory {
  public WebServer getWebServer(ServletContextInitializer... initializers) {
    // ...
    Tomcat tomcat = new Tomcat();
    // ...
    // 預處理上下文
    prepareContext(tomcat.getHost(), initializers);
    return getTomcatWebServer(tomcat);
  }
  protected void prepareContext(Host host, ServletContextInitializer[] initializers) {
    // ...
    // 配置上下文
    configureContext(context, initializersToUse);
  }
  protected void configureContext(Context context, ServletContextInitializer[] initializers) {
    TomcatStarter starter = new TomcatStarter(initializers);
    // ...
    // 在這里就將錯誤的頁面注冊到了tomcat容器中
    for (ErrorPage errorPage : getErrorPages()) {
      org.apache.tomcat.util.descriptor.web.ErrorPage tomcatErrorPage = new org.apache.tomcat.util.descriptor.web.ErrorPage();
      tomcatErrorPage.setLocation(errorPage.getPath());
      tomcatErrorPage.setErrorCode(errorPage.getStatusCode());
      tomcatErrorPage.setExceptionType(errorPage.getExceptionName());
      context.addErrorPage(tomcatErrorPage);
    }
    // ...
  }
}

到此你就知道了一個錯誤的頁是如何在Springboot中被注冊的。到目前為止我們看到的注冊到tomcat容器中的錯誤頁都是個地址,比如:默認是/error。那這個默認的/error又是怎么提供的接口呢?

默認錯誤頁

在Springboot中默認有個自動配置的錯誤頁,在上面有一個代碼片段你應該注意到了

@AutoConfigureBefore(WebMvcAutoConfiguration.class)
@EnableConfigurationProperties({ ServerProperties.class, WebMvcProperties.class })
public class ErrorMvcAutoConfiguration {
  @Bean
  @ConditionalOnMissingBean(value = ErrorAttributes.class, search = SearchStrategy.CURRENT)
  public DefaultErrorAttributes errorAttributes() {
    return new DefaultErrorAttributes();
  }
  @Bean
  @ConditionalOnMissingBean(value = ErrorController.class, search = SearchStrategy.CURRENT)
  public BasicErrorController basicErrorController(ErrorAttributes errorAttributes, ObjectProvider<ErrorViewResolver> errorViewResolvers) {
    return new BasicErrorController(errorAttributes, this.serverProperties.getError(), errorViewResolvers.orderedStream().collect(Collectors.toList()));
  }
}

查看這個Controller

// 默認的錯誤頁地址是/error
@Controller
@RequestMapping("${server.error.path:${error.path:/error}}")
public class BasicErrorController extends AbstractErrorController {
  
  @RequestMapping(produces = MediaType.TEXT_HTML_VALUE)
  public ModelAndView errorHtml(HttpServletRequest request, HttpServletResponse response) {
    HttpStatus status = getStatus(request);
    Map<String, Object> model = Collections.unmodifiableMap(getErrorAttributes(request, getErrorAttributeOptions(request, MediaType.TEXT_HTML)));
    response.setStatus(status.value());
    ModelAndView modelAndView = resolveErrorView(request, response, status, model);
    return (modelAndView != null) ? modelAndView : new ModelAndView("error", model);
  }


  @RequestMapping
  public ResponseEntity<Map<String, Object>> error(HttpServletRequest request) {
    HttpStatus status = getStatus(request);
    if (status == HttpStatus.NO_CONTENT) {
      return new ResponseEntity<>(status);
    }
    Map<String, Object> body = getErrorAttributes(request, getErrorAttributeOptions(request, MediaType.ALL));
    return new ResponseEntity<>(body, status);
  }


}

這里有兩個方法,分別處理了不同的Accept請求頭。到此你是否真正地明白了Springboot中的錯誤處理的工作原理呢?

責任編輯:武曉燕 來源: Spring全家桶實戰案例源碼
相關推薦

2018-09-02 15:43:56

Python代碼編程語言

2021-08-30 07:49:33

索引ICP Mysql

2024-08-02 16:31:12

2023-09-08 08:23:29

Servlet程序MVC

2023-11-30 08:32:31

OpenFeign工具

2024-09-06 17:55:27

Springboot開發

2021-06-03 08:05:46

VSCode 代碼高亮原理前端

2010-08-29 21:09:57

DHCP協議

2023-11-15 08:22:42

Java開發小技巧

2020-06-12 09:20:33

前端Blob字符串

2020-07-28 08:26:34

WebSocket瀏覽器

2024-01-26 06:26:42

Linuxfzf工具

2017-03-13 10:35:10

JavaScript錯誤調用棧

2018-05-17 09:32:52

混合云云計算IT

2024-10-05 00:00:00

HTTPS性能HTTP/2

2024-06-20 08:06:30

2022-04-24 16:00:15

LinuxLinux命令ls命令

2023-12-13 08:28:07

2021-07-14 11:25:12

CSSPosition定位

2010-08-23 09:56:09

Java性能監控
點贊
收藏

51CTO技術棧公眾號

精品三级国产| 中文字幕中文字幕在线中心一区| 91福利精品第一导航| 2019中文字幕在线观看| 九一在线视频| 国产69精品久久99不卡| 亚洲无亚洲人成网站77777| 日韩福利一区二区三区| 玖玖精品在线| 国产精品情趣视频| 国内精品久久久久久| 91免费精品国自产拍在线不卡| 欧美va亚洲va在线观看蝴蝶网| 欧美夫妻性生活| 成人au免费视频影院| 亚洲人成免费| 亚洲网站在线看| 亚洲天天影视| 日韩美女视频一区二区 | 欧美国产日韩精品免费观看| 一本精品一区二区三区| 日本精品视频一区二区| 日本欧洲国产一区二区| 成人97精品毛片免费看| 狠狠躁夜夜躁人人躁婷婷91| 欧美激情视频一区二区三区| 污片视频在线免费观看| 91麻豆精品国产自产在线| 一区二区三区四区视频在线观看 | 日本久久精品一区二区| 韩国午夜理伦三级不卡影院| 国产精品都在这里| 神马电影网我不卡| 欧美精品一区二区在线观看| av国产在线观看| 午夜视频在线观看一区二区三区| 熟女少妇在线视频播放| 亚洲一区二区在线免费观看| 国产成人午夜视频| 缅甸午夜性猛交xxxx| 色天天综合久久久久综合片| 日本欧美色综合网站免费| 欧美日韩尤物久久| 99re亚洲国产精品| 欧美人在线视频| 亚洲国产一区二区三区网站| 亚洲一区亚洲二区| 美女被人操视频在线观看| 欧美优质美女网站| **精品中文字幕一区二区三区| 五月婷婷激情综合网| 亚洲国产成人精品女人久久久 | 成人在线亚洲| 亚洲激情中文字幕| 国产亚洲高清在线观看| 欧美极品欧美精品欧美视频| jizz国产精品| 亚洲性日韩精品一区二区| 女人体1963| 亚洲日本一区二区| 欧美性视频在线播放| 欧美日韩国产丝袜美女| 久久久久九九九| 国产福利不卡视频| 少妇**av毛片在线看| 欧美一级日韩不卡播放免费| 韩国av网站在线| 免费不卡在线观看av| 亚洲午夜久久| 懂色av一区二区三区在线播放| 午夜精品毛片| 秋霞成人午夜鲁丝一区二区三区| 欧美孕妇孕交xxⅹ孕妇交| 人人超碰91尤物精品国产| 亚洲视频sss| 亚洲男人的天堂在线aⅴ视频| 色网址在线观看| 日韩一区二区三| 成人免费短视频| 亚洲毛片一区二区| 四虎成人精品一区二区免费网站| 久久精品国产精品青草色艺 | 97影院秋霞午夜在线观看| 美女三级99| 日韩欧美国产不卡| 西西裸体人体做爰大胆久久久| 美女做a视频| 欧美国产欧美亚洲国产日韩mv天天看完整 | 激情在线视频| 综合五月婷婷| 丝袜老师办公室里做好紧好爽 | 91九色对白| av在线播放成人| 日本天堂影院在线视频| 欧美成人免费在线视频| 亚洲国产婷婷| 国产成人亚洲精品无码h在线| 欧美日韩性生活视频| 精品一区二区免费看| 国内av免费| 成人在线一区| 超级污的网站| 日韩在线观看视频免费| 日韩成人免费看| 黄色一级免费大片| 欧美黑人巨大精品一区二区| 国产成人高清| 欧美xxxx免费虐| 中文字幕中文字幕在线中一区高清 | 欧美在线激情网| 久久电影国产免费久久电影| 自由的xxxx在线视频| 日韩精品在线电影| 国产东北露脸精品视频| 成人不卡视频| 丁香啪啪综合成人亚洲| 成人黄色av| 三级视频网站在线| 久久精品女人的天堂av| 欧美日韩五月天| 第一会所亚洲原创| 国产综合在线观看| 国产在线视频不卡| 亚洲午夜精品网| 国产精品久久久久久久免费观看| 九一免费在线观看| 亚洲精品视频一区| 国产精品无码专区av在线播放| 少妇激情综合网| 亚洲欧美在线高清| 99国产精品视频免费观看一公开| 成人mm视频在线观看| 啊灬啊灬啊灬啊灬高潮在线看| 亚洲人成人77777线观看| 欧美黄网免费在线观看| 91麻豆精品国产91| 亚洲人成精品久久久久久| 91精品一区二区三区综合| 亚洲一区二区少妇| 日韩电视剧免费观看网站| 国产精品国产三级国产专播品爱网| 日韩电影在线一区二区三区| 欧美三级电影网址| 精品99在线视频| 欧美高跟鞋交xxxxxhd| 9191成人精品久久| 亚洲综合成人在线| 成人免费视频一区| 蜜桃在线一区| 在线免费91| 91午夜在线观看| 亚洲va久久久噜噜噜久久狠狠 | 手机av在线| 少妇淫片在线影院| 91九色美女在线视频| 女人被男人躁得好爽免费视频 | 亚洲综合在线一区| 手机看片福利日韩| 99re视频在线| 在线观看亚洲视频| 91丨porny丨首页| 最新中文字幕一区二区三区| 精品久久久久久久久久久久| 久久蜜桃av一区二区天堂| 亚洲国产视频在线| 欧美精品在线视频| 亚洲欧美精品一区| 91免费人成网站在线观看18| 成人污网站在线观看| 同心难改在线观看| 日本乱理伦在线| 精品少妇av| 日本一区二区免费在线 | 蜜桃精品久久久久久久免费影院| 久久久久免费网| av电影一区二区三区| 麻豆一区二区三区四区精品蜜桃| h片精品在线观看| 婷婷久久一区| 国产精品国产三级国产aⅴ无密码| 91精品综合久久久久久| 日韩美女主播视频| 国产精品91在线| 欧美无砖专区免费| 成a人片在线观看| 欧美偷拍综合| 中文字幕va一区二区三区| 日韩三级av在线播放| 国产成人精品在线观看| 国产对白在线播放| 国产一级片在线播放| 精品国产第一国产综合精品| 老牛影视一区二区三区| 亚洲最新视频在线播放| 欧美最猛性xxxxx(亚洲精品)| 成人羞羞国产免费网站| 日韩国产一二三区| 日韩不卡一二三区| 欧美日韩日日摸| 麻豆av一区二区|