经过《基于纯Java代码的Spring容器和Web容器零配置的思考和实现(1) - 数据源与事务管理》和《基于纯Java代码的Spring容器和Web容器零配置的思考和实现(2) - 静态资源、视图和消息器》两篇博文的介绍,我们已经配置好了Spring所需的基本配置。在这边博文中,我们将介绍怎么使用这些配置到实际项目中,并将web.xml文件替换为一个Java类。
我们使用Java代码来配置Spring,目的就是使我们的这些配置能够复用,对于这些配置的复用,我们采用继承和引入来实现。我们同样提供两个类在项目中配置Spring容器。首先提供一个AppConfig类来配置Spring容器,源码如下:
package com.kiiwow; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.EnableAspectJAutoProxy; import org.springframework.context.annotation.Import; import org.springframework.context.annotation.PropertySource; import org.springframework.transaction.annotation.EnableTransactionManagement; import com.kiiwow.framework.config.context.spring.rest.RestSpringContextConfig; import com.kiiwow.framework.config.context.spring.rest.dbconfig.C3p0DataSourceDBConfig; @Configuration @EnableAspectJAutoProxy @EnableTransactionManagement @ComponentScan(basePackages = {"com.kiiwow.*"}) @Import({SpringContextConfig.class, C3p0DataSourceDBConfig.class}) @PropertySource({"classpath:kiiwow.properties"}) public class AppConfig { }
上面的代码使用到了很多注解,我们逐一解释其作用:
- @Configuration,用于表示这个类是一个配置类,用于配置Spring的相关信息
- @EnableAspectJAutoProxy,启用切面自动代理,用于AOP
- EnableTransactionManagement,启用注解事务,即可以使用@Transactional注解来控制事务
- ComponentScan,组件扫描,在basePackages指定的目录下扫描被@Controller、@Service、@Component等注解注册的组件
- @Import,引入指定的配置类,我们引入了Spring容器配置类和数据源事务配置类
- @PropertySource,加载指定的配置文件,配置文件内容会加载入Environment中等待调用
这样我们就把Spring容器所需要的配置完成了,接下来我们提供一个WebConfig类来配置Web子容器的相关内容,源码如下:
package com.kiiwow; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.PropertySource; import org.springframework.web.servlet.config.annotation.EnableWebMvc; import org.springframework.web.servlet.config.annotation.InterceptorRegistry; import com.kiiwow.framework.config.context.spring.WebContextConfig; @Configuration @EnableWebMvc @ComponentScan(basePackages = {"com.kiiwow.*"}) @PropertySource({"classpath:kiiwow.properties"}) public class WebConfig extends WebContextConfig { @Override public void addInterceptors(InterceptorRegistry registry) { //注册自定义监听器 } }
@EnableWebMvc注解用于启用SpringMVC,我们让WebConfig继承自我们已经设计好的Web子容器配置类WebContextConfig,这样WebConfig就已经拥有了所需要的基本配置。由于WebContextConfig本身继承了org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter,所以我们可以直接重写void addInterceptors(InterceptorRegistry registry)方法来注册自定义的监听器。
至此,我们基于Java代码对于Spring的基本配置就全部完成了。但是我们知道,SpringMVC要运行起来还需要在web.xml中配置相关的Servlet,不过既然我们对Spring都采用了全Java配置,那我们也不妨用Java代码来代替web.xml配置。我们提供一个KiiwowInitializer类来替代web.xml,这个类实现了org.springframework.web.WebApplicationInitializer接口,接口中有一个方法void onStartup(ServletContext container) throws ServletException,这个方法的方法体就是web容器启动时需要做的动作,也就是我们以前在web.xml中配置的信息。源码如下:
package com.kiiwow; import java.util.EnumSet; import javax.servlet.DispatcherType; import javax.servlet.ServletContext; import javax.servlet.ServletException; import javax.servlet.ServletRegistration.Dynamic; import org.springframework.web.WebApplicationInitializer; import org.springframework.web.context.ContextLoaderListener; import org.springframework.web.context.support.AnnotationConfigWebApplicationContext; import org.springframework.web.filter.CharacterEncodingFilter; import org.springframework.web.servlet.DispatcherServlet; import com.kiiwow.framework.platform.sqlmapping.SQLMappingInitListener; /** * 用于替代web.xml的web容器配置类 * 在这里配置过滤器、监听器、Servlet * * @author leon.gan * */ public class KiiwowInitializer implements WebApplicationInitializer { @Override public void onStartup(ServletContext container) throws ServletException { //配置Spring提供的字符编码过滤器 javax.servlet.FilterRegistration.Dynamic filter = container.addFilter("encoding", new CharacterEncodingFilter()); //配置过滤器的过滤路径 filter.addMappingForUrlPatterns(EnumSet.of(DispatcherType.REQUEST), true, "/"); //基于注解配置的Spring容器上下文 AnnotationConfigWebApplicationContext rootContext = new AnnotationConfigWebApplicationContext(); //注册Spring容器配置类 rootContext.register(AppConfig.class); container.addListener(new ContextLoaderListener(rootContext)); //SQL配置文件监听器 container.addListener(new SQLMappingInitListener()); //基于注解配置的Web容器上下文 AnnotationConfigWebApplicationContext context = new AnnotationConfigWebApplicationContext(); //注册Web容器配置类 context.register(WebConfig.class); Dynamic servlet = container.addServlet("dispatcher", new DispatcherServlet(context)); //配置映射路径 servlet.addMapping("/"); //启动顺序 servlet.setLoadOnStartup(1); } }
我们首先将Spring提供的字符编码过滤器注册进容器,然后创建基于注解配置的Spring容器上下文,这个context用于注册对Spring容器进行配置的配置类,即刚才我们配置的AppConfig,然后将这个context加入容器的监听器中,使其生命周期与web容器的生命周期一致,然后添加一系列自定义的监听器。接着我们创建基于注解配置的Web容器上下文,这个上下文用于注册对SpringWeb容器进行配置的配置类,即我们刚才定义的WebConfig,然后基于这个上下文创建org.springframework.web.servlet.DispatcherServlet.DispatcherServlet用于拦截和转发请求。
至此,我们所有的配置都已经完成,这个时候我们项目中对于配置这一块就不会再存在XML文件了。另外,如果我们是使用Maven管理项目,这个时候Maven找不到我们项目中的web.xml文件就会在POM文件报错,并且也无法进行打包操作。这时我们需要在POM文件中添加一个WAR包插件用于抑制这个错误,源码如下:
<plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-war-plugin</artifactId> <version>2.2</version> <configuration> <failOnMissingWebXml>false</failOnMissingWebXml> </configuration> </plugin>
经过三篇博文,我们介绍了基于纯Java代码的Spring容器和Web容器零配置的方法,希望能够在Spring的使用上对大家有所帮助。感谢您的阅览,麻烦点个赞。