SpringBoot默认是用的是Tomcat作为嵌入式的Servlet容器;
问题?
1)、如何定制和修改Servlet容器的相关配置;
1、修改和server有关的配置(ServerProperties):
1 2 3 4 5 6 7 8 | server.port= 8081 server.context-path=/crud server.tomcat.uri-encoding=UTF- 8 //通用的Servlet容器设置 server.xxx //Tomcat的设置 server.tomcat.xxx |
2、编写一个EmbeddedServletContainerCustomizer:嵌入式的Servlet容器的定制器;来修改Servlet容器的配置;
1 2 3 4 5 6 7 8 9 10 11 | @Bean public EmbeddedServletContainerCustomizer embeddedServletContainerCustomizer(){ return new EmbeddedServletContainerCustomizer() { //定制嵌入式的Servlet容器相关的规则 @Override public void customize(ConfigurableEmbeddedServletContainer container) { container.setPort( 8083 ); } }; } |
2)、注册Servlet三大 组件(Servlet、Filter、Listener)
由于SprintBoot默认是以jar包的方式启动嵌入式的Servlet容器来启动SpringBoot的web应用,没有web.xml文件。
注册三大组件用以下方式
1 2 3 | ServletRegistrationBean FilterRegistrationBean ServletListenerRegistrationBean |
SpringBoot帮我们自动SpringMVC的时候,自动的注册SpringMVC的前端控制器;DispatcherServlet;
//默认拦截: / 所有请求;包静态资源,但是不拦截jsp请求; /*会拦截jsp
//可以通过server.serverPath来修改SpringMVC前端控制器默认拦截的请求路径;
2)、SpringBoot能不能支持其他的Servlet容器;
3)、替换为其它嵌入式Servlet容器
默认支持
Tomcat(默认使用);
Jetty
Undertow
4)、嵌入式Servlet容器自动配置原理;
EmbeddedServletContainerAutoConfiguration:配置签入是的Servlet容器自动配置? @AutoConfigureOrder(-2147483648) @Configuration @ConditionalOnWebApplication @Import({EmbeddedServletContainerAutoConfiguration.BeanPostProcessorsRegistrar.class})//导入BeanPostProcessorsRegistrar:Spring注解版;给容器中导入一些组件; //导入了embeddedServletContainerCustomizerBeanPostProcessor; //后置处理器:bean初始化前后(创建完对象,还没属性赋值)执行初始化工作 public class EmbeddedServletContainerAutoConfiguration { @Configuration @ConditionalOnClass({Servlet.class, Tomcat.class})//判断当前是否引入了Tomcat依赖; @ConditionalOnMissingBean( value = {EmbeddedServletContainerFactory.class}, search = SearchStrategy.CURRENT )//判断当前容器没有用户自己定义EmbeddedServletContainerFactory;嵌入式的Servlet容器工厂;作用:创建嵌入式的Servlet容器; public static class EmbeddedTomcat { public EmbeddedTomcat() { } @Bean public TomcatEmbeddedServletContainerFactory tomcatEmbeddedServletContainerFactory() { return new TomcatEmbeddedServletContainerFactory(); } }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 | 1 )、EmbeddedServletContainerFactory(嵌入式Servlet容器工厂) public interface EmbeddedServletContainerFactory { //获取嵌入式的Servlet容器 EmbeddedServletContainer getEmbeddedServletContainer(ServletContextInitializer... var1); } 2 )、EmbeddedServletContainer (嵌入式的Servlet容器) 3 )、以TomcatEmbeddedServletContainerFactory为例 public EmbeddedServletContainer getEmbeddedServletContainer(ServletContextInitializer... initializers) { //创建一个Tomcat Tomcat tomcat = new Tomcat(); File baseDir = this .baseDirectory != null ? this .baseDirectory : this .createTempDir( "tomcat" ); //配置Tomcat的基本环节 tomcat.setBaseDir(baseDir.getAbsolutePath()); Connector connector = new Connector( this .protocol); tomcat.getService().addConnector(connector); this .customizeConnector(connector); tomcat.setConnector(connector); tomcat.getHost().setAutoDeploy( false ); this .configureEngine(tomcat.getEngine()); Iterator var5 = this .additionalTomcatConnectors.iterator(); while (var5.hasNext()) { Connector additionalConnector = (Connector)var5.next(); tomcat.getService().addConnector(additionalConnector); } this .prepareContext(tomcat.getHost(), initializers); //将配置好的Tomcat传入进去,返回一个EmbeddedServletContainer ;并且启动Tomcat服务器; return this .getTomcatEmbeddedServletContainer(tomcat); } 4 )、我们对嵌入式容器的配置修改是怎么生效? ServerProperties、EmbeddedServletContainerCustomizer EmbeddedServletContainerCustomizer:定制器帮我们修改了Servlet容器的配置? 怎么修改的原理? 5 )、容器中导入了embeddedServletContainerCustomizerBeanPostProcessor //初始化之前 public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { //如果当前初始化的是一个ConfigurableEmbeddedServletContainer类型的 组件 if (bean instanceof ConfigurableEmbeddedServletContainer) { this .postProcessBeforeInitialization((ConfigurableEmbeddedServletContainer)bean); } return bean; } private void postProcessBeforeInitialization(ConfigurableEmbeddedServletContainer bean) { Iterator var2 = this .getCustomizers().iterator(); //获取所有的定制器,调用每一个定制器的customize方法来给Servlet容器进行属性赋值; while (var2.hasNext()) { EmbeddedServletContainerCustomizer customizer = (EmbeddedServletContainerCustomizer)var2.next(); customizer.customize(bean); } } private Collection<EmbeddedServletContainerCustomizer> getCustomizers() { if ( this .customizers == null ) { //从容器中获取所有这个类型的组件:EmbeddedServletContainerCustomizer //定制Servlet容器,给容器中可以添加一个EmbeddedServletContainerCustomizer类型的组件; this .customizers = new ArrayList( this .beanFactory.getBeansOfType(EmbeddedServletContainerCustomizer. class , false , false ).values()); Collections.sort( this .customizers, AnnotationAwareOrderComparator.INSTANCE); this .customizers = Collections.unmodifiableList( this .customizers); } return this .customizers; } ServerProperties也是定制器; 步骤: 1 )、SpringBoot根据导入的依赖情况,给容器中添加相应的EmbeddedServletContainerFactory【TomcatEmbeddedServletContainerFactory】 2 )、容器中某个组件要创建对象就会惊动后置处理器; EmbeddedServletContainerCustomizerBeanPostProcessor: 只要是嵌入式的Servlet容器工厂,后置处理器就工作; 3 )、后置处理器,从容器中获取所有的EmbeddedServletContainerCustomizer,调用定制器的方法; 5 )、嵌入式Servlet容器启动原理; 什么时候创建嵌入式的Servlet容器工厂?什么时候获取嵌入式的Servlet容器并启动Tomcat; 获取嵌入式的Servlet容器工厂: 1 )、SpringBoot应用启动运行run方法 2 )、refreshContext(context);SpringBoot刷新IOC容器(创建IODC容器对象,并初始化容器,创建容器中的每一个组件);如果是web应用创建AnnotationConfigEmbeddedWebApplicationContext,否则创建AnnotationConfigApplicationContext 3 )、refresh(context);刷新刚才创建好的ioc容器; 4 )、onRefresh();web的ioc容器重写了onFresh()方法 5 )、webioc容器会创建嵌入式的Servlet容器;createEmbeddedServletContainer(); 6 )、获取嵌入式的 Servlet容器工厂; 从ioc容器中获取EmbeddedServletContainerFactory创建对象,后置处理器一看是这个对象,就获取 所有的定制器来先定制Servlet容器的相关配置; 7 )、使用容器工厂获取嵌入式的Servlet容器 8 )、嵌入式的Servlet容器创建对象并启动Servlet容器; 先启动嵌入式的Servlet容器,再将ioc容器中剩下没有创建出的对象获取处出来; IOC容器启动创建嵌入式Servlet容器; |
用代码行来衡量开发进度,无异于用重量来衡量制造飞机的进度。
【推荐】还在用 ECharts 开发大屏?试试这款永久免费的开源 BI 工具!
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步