宁可五年学成一技,也不要一年学五技。能够积累才是多,不用重来才是快
【小家Spring】由Spring注解驱动开发引发的疑问:ServletContainerInitializer加载机制,以及ServiceLoader的使用(以JDBC为例介绍SPI)
【小家Spring】Spring MVC控制器中Handler的四种实现方式:Controller、HttpRequestHandler、Servlet、@RequestMapping
诚如各位所知,Servlet3.0是一次Java EE规范的一次重要升级。支持到可以全部采用注解驱动,大大简化了配置web.xml的麻烦。现在启动一个web容器并不强制依赖于web.xml部署描述符了。
然后我们印象深刻的是,之前我们在使用Spring MVC的时候,DispatcherServlet
是必须要要在web.xml里配置,现在没有了这个,我们该怎么办呢?
本文主要以全注解驱动整合Spring MVC(注意:非Spring Boot环境,否则内部细节都看不到了)抛出问题,从而从内部原理方面去了解里面的门道。
Spring源码基于的Spring版本为:5.0.6.RELEASE(下同)
Spring源码基于的Spring版本为:5.0.6.RELEASE(下同)
Spring源码基于的Spring版本为:5.0.6.RELEASE(下同)
准备一个Spring MVC的maven工程
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0modelVersion>
<groupId>com.fsxgroupId>
<artifactId>demo-warartifactId>
<version>1.0-SNAPSHOTversion>
<packaging>warpackaging>
<properties>
<project.build.sourceEncoding>UTF-8project.build.sourceEncoding>
<project.reporting.outputEncoding>UTF-8project.reporting.outputEncoding>
<java.version>1.8java.version>
properties>
<dependencies>
<dependency>
<groupId>org.springframeworkgroupId>
<artifactId>spring-webmvcartifactId>
<version>5.0.6.RELEASEversion>
dependency>
<dependency>
<groupId>javax.servletgroupId>
<artifactId>javax.servlet-apiartifactId>
<version>3.1.0version>
<scope>providedscope>
dependency>
<dependency>
<groupId>org.projectlombokgroupId>
<artifactId>lombokartifactId>
<scope>providedscope>
<version>1.18.4version>
dependency>
<dependency>
<groupId>ch.qos.logbackgroupId>
<artifactId>logback-classicartifactId>
<version>1.2.3version>
dependency>
<dependency>
<groupId>com.fasterxml.jackson.coregroupId>
<artifactId>jackson-databindartifactId>
<version>2.9.8version>
dependency>
<dependency>
<groupId>com.alibabagroupId>
<artifactId>fastjsonartifactId>
<version>1.2.57version>
dependency>
dependencies>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.pluginsgroupId>
<artifactId>maven-war-pluginartifactId>
<version>2.6version>
<configuration>
<failOnMissingWebXml>falsefailOnMissingWebXml>
configuration>
plugin>
<plugin>
<groupId>org.apache.maven.pluginsgroupId>
<artifactId>maven-compiler-pluginartifactId>
<version>3.8.0version>
<configuration>
<source>${java.version}source>
<target>${java.version}target>
<compilerVersion>${java.version}compilerVersion>
<encoding>${project.build.sourceEncoding}encoding>
configuration>
plugin>
plugins>
<resources>
<resource>
<directory>src/main/javadirectory>
<includes>
<include>**/*.propertiesinclude>
<include>**/*.xmlinclude>
<include>**/*.tldinclude>
includes>
<filtering>falsefiltering>
resource>
resources>
build>
project>
然后个logback.xml
一个最小配置:
<configuration scan="true" scanPeriod="60 seconds" debug="false">
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%npattern>
encoder>
appender>
<root level="INFO">
<appender-ref ref="STDOUT"/>
root>
configuration>
备注:运行本war包的web容器为:tomcat-8.0(最高支持到了Servlet3.1~)
写一个最基本的Servlet,然后就可以访问了http://localhost:8080/demowar_war/hello
:
/**
* @author fangshixiang
* @description
* @date 2019-02-16 22:04
*/
@WebServlet(urlPatterns = "/hello")
public class HelloServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.getWriter().write("hello servlet...");
}
}
我们发现,比之前采用web.xml配置的方式,省事太多了,可谓非常方便。
之前web容器要整合其余模块,都是通过web.xml来的。那么现在注解驱动的话,怎么做呢?
这就是Servlet3.0给我们提供的特别特别重要的一个类ServletContainerInitializer
来整个其它模块组件。通过读Servlet3.0的官方文档如下:
大致可以看出如下意思,它有如下能力:
Shared libraries(共享库) / runtimes pluggability(运行时插件能力)
META-INF/services/javax.servlet.ServletContainerInitializer
这个文件里,文件内容为就是ServletContainerInitializer
实现类的全类名;这样web容器在启动的时候,就会执行该接口的实现方法,从而我们就可以书写我们自己的模块初始化的一些逻辑。
//容器启动的时候会将@HandlesTypes指定的这个类型下面的子类(实现类,子接口等)传递过来;
@HandlesTypes(value = {HelloService.class})
public class MyServletContainerInitializer implements ServletContainerInitializer {
/**
* 应用启动的时候,会运行onStartup方法;
*
* Set> c:感兴趣的类型的所有子类型;
* ServletContext ctx:代表当前Web应用的ServletContext;一个Web应用一个ServletContext;
*
*/
@Override
public void onStartup(Set<Class<?>> c, ServletContext ctx) throws ServletException {
//这里的c会把所有我们感兴趣的类型都拿到
System.out.println("感兴趣的类型:");
for (Class<?> claz : c) {
System.out.println(claz);
}
//==========================编码形式注册三大组件============================
////注册组件 ServletRegistration
//ServletRegistration.Dynamic servlet = ctx.addServlet("userServlet", new UserServlet());
////配置servlet的映射信息
//servlet.addMapping("/user");
//
////注册Listener
//ctx.addListener(UserListener.class);
//
////注册Filter FilterRegistration
//FilterRegistration.Dynamic filter = ctx.addFilter("userFilter", UserFilter.class);
////配置Filter的映射信息
//filter.addMappingForUrlPatterns(EnumSet.of(DispatcherType.REQUEST), true, "/*");
}
}
启动容器,我们会看到:
把我们关心的接口子类型(包含子接口、抽象类、实现类)都放进来了。但是需要注意:不包含自己哦~
/不拦截.jsp。而/*都会拦截
DispatcherServlet映射:/
Filter映射:/*
Spring MVC拦截器的映射:/**
整合Spring MVC是重中之重。前面已经说过ServletContainerInitializer
了,相信大家能够想到Spring是怎么做的了吧?直接参照Spring官方文档先看看:
链接如下:https://docs.spring.io/spring/docs/5.1.5.RELEASE/spring-framework-reference/web.html#mvc-servlet
我们看看Spring-web包jar包内:
很显然的发现,Spring MVC也是通过这种方式和Servlet容器进行整合的。web容器在启动的时候,就会自动去加载org.springframework.web.SpringServletContainerInitializer
这个类。源码如下(它关心的是WebApplicationInitializer
的子类们):
@HandlesTypes(WebApplicationInitializer.class)
public class SpringServletContainerInitializer implements ServletContainerInitializer {
@Override
public void onStartup(@Nullable Set<Class<?>> webAppInitializerClasses, ServletContext servletContext)
throws ServletException {
List<WebApplicationInitializer> initializers = new LinkedList<>();
if (webAppInitializerClasses != null) {
for (Class<?> waiClass : webAppInitializerClasses) {
// Be defensive: Some servlet containers provide us with invalid classes,
// no matter what @HandlesTypes says...
if (!waiClass.isInterface() && !Modifier.isAbstract(waiClass.getModifiers()) &&
WebApplicationInitializer.class.isAssignableFrom(waiClass)) {
try {
initializers.add((WebApplicationInitializer)
ReflectionUtils.accessibleConstructor(waiClass).newInstance());
}
catch (Throwable ex) {
throw new ServletException("Failed to instantiate WebApplicationInitializer class", ex);
}
}
}
}
if (initializers.isEmpty()) {
servletContext.log("No Spring WebApplicationInitializer types detected on classpath");
return;
}
servletContext.log(initializers.size() + " Spring WebApplicationInitializers detected on classpath");
AnnotationAwareOrderComparator.sort(initializers);
for (WebApplicationInitializer initializer : initializers) {
initializer.onStartup(servletContext);
}
}
}
步骤分析:
说明:虽然父类只有两个
abstract
抽象方法要求子类必须实现。但是父类的设计都是可以扩展的,若你想定制化自己的需求,都是可以通过复写父类的protected的方式扩展的
比如:你想定制化自己的DispatcherServlet
(父类默认值是单纯的new一下),那么你就可以通过复写createDispatcherServlet()
去定制~
从上面分析得知,我们要使用注解驱动的话。只需要我们自己实现AbstractAnnotationConfigDispatcherServletInitializer
这个抽象类就行了,这样web容器启动的时候就能处理我们实现的这个类的内容。示例如下(采用父子容器):
父子容器的配置类:
// 备注:此处@ControllerAdvice、RestControllerAdvice 这个注解不要忘了,属于Controller层处理全局异常的,应该交给web去扫描
@ComponentScan(value = "com.fsx", excludeFilters = {
@Filter(type = FilterType.ANNOTATION, classes = {Controller.class, ControllerAdvice.class, RestControllerAdvice.class})
})
@Configuration //最好标注上,本人亲测若不标准,可能扫描不生效
public class RootConfig {
}
// 此处记得排除掉@Controller和@ControllerAdvice、@RestControllerAdvice
@ComponentScan(value = "com.fsx", useDefaultFilters = false,
includeFilters = {@Filter(type = FilterType.ANNOTATION, classes = {Controller.class, ControllerAdvice.class, RestControllerAdvice.class})}
)
@Configuration //最好标注上,本人亲测若不标准,可能扫描不生效
public class AppConfig {
}
@RestControllerAdvice
是Spring4.3后提供的注解。@ControllerAdvice是Spring3.2提供的
它俩的区别就像@Controller和@RestController的区别。(也就是说@RestControllerAdvice``可以省略
@ResponseBody`不用写了~~~)
useDefaultFilters默认值为true,表示默认情况下@Component、@Repository、@Service、@Controller都会扫描
useDefaultFilters=false加上includeFilters我们就可以只扫描指定的组件了,比如Spring MVC的web子容器只扫描Controller组件
excludeFilters的时候,就不需要去设置useDefaultFilters=false,这样子我们直接排除掉即可哟~
特别注意
:useDefaultFilters
的正确使用,不要造成重复扫描。否则很有可能造成事务不生效
,并且你还非常不好定位这个错误。然后我们自己来实现AbstractAnnotationConfigDispatcherServletInitializer
一个初始化实体类:
/**
* 自己实现 基于注解驱动的ServletInitializer来初始化DispatcherServlet
*/
public class MyWebAppInitializer extends AbstractAnnotationConfigDispatcherServletInitializer {
/**
* 根容器的配置类;(Spring的配置文件) 父容器;
*/
@Override
protected Class<?>[] getRootConfigClasses() {
return new Class<?>[]{RootConfig.class};
}
/**
* web容器的配置类(SpringMVC配置文件) 子容器;
*/
@Override
protected Class<?>[] getServletConfigClasses() {
return new Class<?>[]{AppConfig.class};
}
//获取DispatcherServlet的映射信息
// 注意: /:拦截所有请求(包括静态资源(xx.js,xx.png)),但是不包括*.jsp;
// /*:拦截所有请求;连*.jsp页面都拦截;jsp页面是tomcat的jsp引擎解析的;
@Override
protected String[] getServletMappings() {
return new String[]{"/"};
}
// 若你想定制化父类的一些默认行为 这里都是可以复写父类的protected方法的~~~~
// Spring MVC也推荐你这么干~
@Override
protected FrameworkServlet createDispatcherServlet(WebApplicationContext servletAppContext) {
DispatcherServlet dispatcherServlet = (DispatcherServlet) super.createDispatcherServlet(servletAppContext);
// dispatcherServlet.setDetectAllHandlerAdapters(false);
return dispatcherServlet;
}
}
我们写个测试类试试:
@Controller
public class HelloController {
@Autowired
HelloService helloService;
@ResponseBody
@RequestMapping("/hello")
public String hello() {
System.out.println(helloService); //com.fsx.service.HelloServiceImpl@512663b0
return "hello...";
}
}
这样我们就可以正常访问controller的请求了。大功告成~
web容器中的Spring的应用 一启动就会 加载感兴趣的
WebApplicationInitializer
接口的下的所有组件,并且为WebApplicationInitializer组件创建对象(组件不是接口,不是抽象类)。
按照上面的配置,我偶然的发现了,RootConfig仍然还是去扫描了我的controller,导致我的controller被扫描了两次,怎么回事呢???
找了好久,终于找到原因了,并不是@ComponentScan
或者excludeFilters
的问题,而是因为咱们在执行RootConfig
的时候,虽然不去扫描Controller注解了,但是它会扫描AppConfig.java
这个配置类,从而间接的又去扫描了@Controller
了,因此最正确的做法应该如下:
@ComponentScan(value = "com.fsx", excludeFilters = {
@Filter(type = FilterType.ANNOTATION, classes = {Controller.class}),
//排除掉web容器的配置文件,否则会重复扫描
@Filter(type = FilterType.ASSIGNABLE_TYPE, classes = {AppConfig.class})
})
@Configuration
public class RootConfig {
}
这样子,我们的Controller就只会被扫描一次了,容器也就非常的干净了,强烈建议这么干。
当然,如果你还是当初xml的方式来做的,分别是两个配置文件下,只要不自己import之类,就不会出现此种问题~
容器完全隔离后的好处是非常明显的,比如我们的web组件,就放在AppConig里,其它的放在Root里,不要什么都往RootConfig里面塞,比如如下:
//web子容器里注册一个Child
public class AppConfig {
@Bean
public Child child() {
return new Child();
}
}
//父容器里注册一个Parent
public class RootConfig {
@Bean
public Parent parent() {
return new Parent();
}
}
然后我们会发现在Controller层注入这两个Bean是正常的:
但是在Service层注入,启动的时候就会报错了
报错如下:
这里面说个结论:
1、父子容器的关系就行内部类的关系一样。子容器能得到父容器的Bean,但是父容器得不到子容器的Bean
2、父子容器中,属性值都不是互通的。@Value注入的时候需要注意一下子~
之前我们使用xml文件的时候,我们可以配置Spring MVC等相关选项。
比如视图解析器、视图映射、静态资源映射、拦截器。。。
首先:在配置文件里加上注解@EnableWebMvc
:开启SpringMVC定制配置功能;
其次: 实现WebMvcConfigurer
接口。通过这个接口我们可以发现,里面有很多方法,但大多数情况下我们并不需要配置这么多项,因此Spring MVC也考虑到了这一点,提供给我们一个WebMvcConfigurerAdapter
来extends
就行,Adapter都是空实现~,这样我们需要配置什么,复写对应方法就行
从上可以看出,如果你的项目是构建在Spring5.0(基于java8)以上的,直接实现接口即可。不用再继承此Adapter了~
@EnableWebMvc
@Configuration //一定要说明这个文件是个配置文件
public class WebMvcConfig implements WebMvcConfigurer {
//视图解析器
@Override
public void configureViewResolvers(ViewResolverRegistry registry) {
//默认所有的页面都从 /WEB-INF/ xxx .jsp
//registry.jsp();
registry.jsp("/WEB-INF/views/", ".jsp");
}
// 开启静态资源的请求转发到默认servlet上,不配置页面报错404,(默认servlet不是DispatcherServlet!理解的)
@Override
public void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
configurer.enable();
}
//自定义添加拦截器=========这个比较常用
@Override
public void addInterceptors(InterceptorRegistry registry) {
//registry.addInterceptor(new MyFirstInterceptor()).addPathPatterns("/**");
}
}
这样我们就可以通过此配置文件,个性化定制我们的Spring MVC了。
最后需要多一句嘴,我们还能看到还有一个类:WebMvcConfigurationSupport
。小伙伴们查看很多文章,但此处我只推荐一个老铁的文章,说到了点上:WebMvcConfigurationSupport与WebMvcConfigurer的关系
结论可以摆在此处:最佳实践还是继承WebMvcConfigurerAdapter(或直接实现接口WebMvcConfigurer),只不过要多加一个@EnableWebMvc注解而已。
备注:若是SpringBoot环境,请不要加@EnableWebMvc
注解,因为springboot已经实例化了WebMvcConfigurationSupport,如果添加了该注解,默认的WebMvcConfigurationSupport配置类是不会生效的
WebRequestInterceptor间接实现了HandlerInterceptor,只是他们之间使用WebRequestHandlerInterceptorAdapter适配器类联系。
这两个Spring MVC的拦截器接口比较就比较简单了。直接给结论吧:
最佳实践:HandlerInterceptor
能够实现所有WebRequestInterceptor
做的事,更偏底层些。因此建议使用HandlerInterceptor
最后需要注意一点的是,关于自定义视图解析的自定义配置。此处还有一种方法是直接向容器里面注册Bean即可,如下:
//自定义一个视图解析器
@Bean
public ViewResolver viewResolver() {
InternalResourceViewResolver resolver = new InternalResourceViewResolver();
resolver.setPrefix("WEB-INF/views/");
resolver.setSuffix(".html");
resolver.setExposeContextBeansAsAttributes(true);
return resolver;
}
关于其中缘由原理。接下来相关文章讲解到Spring MVC深层原理剖析的时候,会精讲这一块,有兴趣的小伙伴可以持续关注
由于Spring Boot环境下并不建议启用
@EnableWebMvc
,所以使用起来请自行举一反三
在web.xml中,我们知道,执行顺序是谁在前边执行谁。但是现在没有这个web.xml了,肿么定这个执行顺序呢?
若是在Spring Boot环境,我们很好的确定Bean的执行顺序,我们可以用@Order
注解:
@Bean
@Order(Integer.MAX_VALUE)
也可这么来:
registration.setOrder(Integer.MAX_VALUE);
Spring boot 会按照order值的大小,从小到大的顺序来依次过滤。也就是说,数字越小,越先执行
那么问题来了,现在我们只根据@WebFilter
来排序Filter的执行顺序,怎么破呢?很多人曾经给出答案说没办法,是无序的。
其实不然,经过我的实践发现,servlet容器是按照Filter的类名按照自然顺序排序的。什么意思呢?比如我有两个Filter:UserLoginFilter和ApiLog
。因为这两个文件的首字母A排U之前,所以每次都会限制性ApiLog。
那么我们就是想先要执行UserLoginFilter
怎么办呢?这里有个小技巧,我们可以这么来写即可:
Filter0_UserLogin.java
Filter1_ApiLog.java
完美。但是在Spring Boot环境下的话,还是用Order来控制哦,更加优雅~
从Servlet3.0开始,Spring3.2开始,就推荐全部使用注解来驱动应用了。在当下流行的SpringBoot环境中,注解驱动可以说体现的淋漓尽致,完全摒弃了之前的xml配置文件,化简为繁。
Spring Boot中集成和使用Spring MVC会方便得多得多,因为它都已经帮我们配置好了,但理解了这篇文章的原理,再去理解Boot,可谓就非常顺畅了~
希望本文能帮助到大家理解web容器对Spring MVC的集成。(Spring Boot不同的地方在于它是Spring容器驱动web容器(默认情况下)。而本文是web容器驱动Spring容器)
启动Spring容器有三种方式:我这里推荐这篇文章:spring容器启动的三种方式
需要说明的是:
若我们采用原始的方式,配置ContextLoaderListener
监听器启动的时候会创建一个web容器,部分源代码如下:
可以看出,它配置的是一个Web容器。这样我们的Spring容器就启动了~
关于DispatchServlet
的配置,只是让支持了Spring MVC的功能,能够分发请求了。
今天有同事问”Spring Boot依赖的Embedded的Tomcat能不能同时监听多个端口?“
在回答这个问题之前,体验一把处处留心皆学问
:
学问就在这个s
,说明SpringBoot铁定是支持监听多个端口的~~~~
先看看单体的Tomcat容器:stand-alone的tomcat当然是可以的。Tomcat的架构中,一个Connecter监听一个端口。 如果是stand-alone的Tomcat,只需要在server.xml中添加一个即可,
<Connector port="8080" redirectPort="8443" acceptCount="100" debug="0" connectionTimeout="20000" />
<Connector port="9090" redirectPort="8443" acceptCount="100" debug="0" connectionTimeout="20000" />
上面配置了3个Connector,分别监听 8080,9090这两个个端口。Tomcat启动日志里也可以看出端倪
再看看SpringBoot的嵌入式容器:它使用了使用了Embedded Tomcat
。同时提供了EmbeddedServletContainerCustomizer
接口让用户对各种EmbeddedServletContainer
进行配置。因此我们可以加上如下配置:
@Configuration
public class ServerConfig implements EmbeddedServletContainerCustomizer {
@Override
public void customize(ConfigurableEmbeddedServletContainer container) {
// 此处只处理Tomcat类型的嵌入式容器
if (container instanceof TomcatEmbeddedServletContainerFactory) {
TomcatEmbeddedServletContainerFactory tomcat = (TomcatEmbeddedServletContainerFactory) container;
// 此处一般是读取配置文件~~~此处我就偷懒了~~~
//String[] portsArray = ports.split(",");
String[] portsArray = {"7070", "9090"};
for (String portStr : portsArray) {
int port = Integer.parseInt(portStr);
// Tomcat中,一个Connecter监听一个端口 指定协议为HTTP/1.1
Connector httpConnector = new Connector("HTTP/1.1");
httpConnector.setPort(port);
// 添加一个额外的端口 和server.port不冲突~
tomcat.addAdditionalTomcatConnectors(httpConnector);
}
}
}
}
我们的SpringBoot应用就监听着三个端口了~~~完美
这个也是扩展内容。我们知道我们自己new一个上下文也是ok的。
但是如果我们要new一个web上下文呢?比如我想new一个AnnotationConfigWebApplicationContext
public static void main(String[] args) {
ApplicationContext applicationContext = new AnnotationConfigApplicationContext(RootConfig.class);
System.out.println(applicationContext.containsBean("person")); // true
}
这样Spring上下文就正常启动了。
请注意:若构造函数没有放入Config文件,而是后期自己register进去的,那么请手动refresh()。因为空构造函数式不会自动refresh的
那如果我想一个web环境呢?比如我想new一个 AnnotationConfigWebApplicationContext
。首先我们看看该类源码:它只有一个的构造函数。 若我们模仿着这么做:
public static void main(String[] args) {
AnnotationConfigWebApplicationContext webApplicationContext = new AnnotationConfigWebApplicationContext();
//webApplicationContext.setParent(applicationContext);
webApplicationContext.register(WebMvcConfig.class); // 注册上web环境的的配置类
webApplicationContext.refresh(); // 手动刷新
System.out.println(webApplicationContext.containsBean("helloController "));
}
如果这样,你会看到报错:
Exception in thread "main" java.lang.NoClassDefFoundError: javax/servlet/ServletRequest
没错。这属于Servlet的web组件,它一般都在tomcat等容器里面,我们maven自己导入一般也会规定为
。
好即使我们把scope去掉真的导入进来,再运行依然报错:
threw exception; nested exception is java.lang.IllegalStateException: No ServletContext set
显然web容器的初始化,它是依赖于Servlet上下文的,而我们并没有初始化掉这个上下文,所以就报错了。那我们自己new一个上下文???
到此打住吧~~~如果真的对main方法启动一个web上下文,我强烈建议你关注后面我讲述的关于SpringBoot的启动原理分析,它就是这么来干的,而此处仅仅只是牛刀小试一把~ 毕竟我们还只研究纯Spring环境而非Boot环境~
若群二维码失效,请加微信号(或者扫描下方二维码):fsx641385712。
并且备注:“java入群” 字样,会手动邀请入群