二、Java Config技能

自从Spring MVC3.0技能推出以来,Spring MVC新增了很多Java Config表明,简化了配置,初步构建了无xml启动。

@Configuration

在类上打上这一标签,表示这个类是配置类

config详解JSP从JavaConfig到SpringBoot启动源码道理解析 jQuery

@ComponentScan

相称于Spring MVC中的Component-Scan包扫描Package技能

@EnableWebMvc

相称于Spring MVC xml配置中的<mvc:annotation-driven>

@ImportResource

相称于导入配置文件<import bean-class=&#34;"/>

@PropertySource

用于读取properties属性文件

2.1利用方法

@Configuration

public class AutoConfiguration {

@Bean

public JText button() {

return new JText("Hello World");

}

@Bean

public JButton anotherButton(Icon icon) {

return new JButton(icon);

}

@Bean

public Image autoLogo() throws URLException {

URL url = new URL(

"http://luowei.com/assets/images/luowei_icon.png");

return new ImageIcon(url);

}

}

用法非常大略:

1.用@Configuration表明JavaConfig类,

2.用每个方法来表示Bean并利用@Bean表明方法。

3.每个方法名代表XML配置文件中的name

三、利用JavaConfig代替Spring MVC

创建一个类实现 这个类: WebApplicationInitializer

官方示例代码

public class MyWebApplicationInitializer implements WebApplicationInitializer {

@Override

public void onStartup(ServletContext servletCxt) {

// Load Spring web application configuration

AnnotationConfigWebApplicationContext ac = new AnnotationConfigWebApplicationContext();

ac.register(AppConfig.class);

ac.refresh();

// Create and register the DispatcherServlet

DispatcherServlet servlet = new DispatcherServlet(ac);

ServletRegistration.Dynamic registration = servletCxt.addServlet("app", servlet);

registration.setLoadOnStartup(1);

registration.addMapping("/app/");

}

}

这样就能够在Tomcat中启动DispatcherServlet,从而简化了Spring MVC XML中常化的配置。

3.1自我实现

//相称于web.xml

public class WebInitializer implements WebApplicationInitializer {

public void onStartup(ServletContext servletContext) throws ServletException {

AnnotationConfigWebApplicationContext ownerCtx = new AnnotationConfigWebApplicationContext();

ownerCtx .register(Config.class);//注册mvc的配置类

ownerCtx .setServletContext(servletContext);

/

注册DispatcherServlet(调度事情,掌握流程)

/

Dynamic servlet = servletContext.addServlet("dispatcher", new DispatcherServlet(ownerCtx ));//配置DispatcherServlet

servlet.addMapping("/");//url-pattern路径

servlet.setLoadOnStartup(1);//启动顺序

// 设置编码格式

CharacterEncodingFilter characterEncodingFilter = new CharacterEncodingFilter();

characterEncodingFilter.setEncoding("UTF-8");// 国际化

characterEncodingFilter.setForceEncoding(true);

javax.servlet.FilterRegistration.Dynamic filter = servletContext.addFilter("encoding", characterEncodingFilter);

filter.addMappingForUrlPatterns(EnumSet.allOf(DispatcherType.class),true,"");

}

}

实现JavaConfig:

@Configuration//相称于配置文件

@ComponentScan("com.hdicloud")//配置包扫描

@EnableWebMvc//启用mvc支持 annotation-driven

@MapperScan(value="com.base.lc.mapper",sqlSessionFactoryRef = "sqlSessionFactory")//mapper扫描器

@EnableTransactionManagement//开缘由务

public class SpringConfig extends WebMvcConfigurerAdapter {

/

jsp视图解析器的bean

/

@Bean

public UrlBasedViewResolver setupViewResolver() {

UrlBasedViewResolver resolver = new UrlBasedViewResolver();

resolver.setPrefix("/");

resolver.setSuffix(".jsp");

resolver.setViewClass(JstlView.class);

return resolver;

}

/

配置数据源

/

@Bean(name = "dataSource")

public DriverManagerDataSource getDataSource() {

try {

DriverManagerDataSource dataSource = new DriverManagerDataSource();

dataSource.setDriverClassName("com.mysql.cj.jdbc.Driver");

dataSource.setUrl("jdbc:mysql://192.168.7.9:3306/config?useAffectedRows=true");

dataSource.setUsername("root");

dataSource.setPassword("123456");

return dataSource;

} catch (Exception e) {

return null;

}

}

/

创建事务管理 用来指天命据源

/

@Bean

public DataSourceTransactionManager transactionManager() {

return new DataSourceTransactionManager(getDataSource());

}

/

配置SqlSessionFactory,给MapperScan调用

/

@Bean

public SqlSessionFactory sqlSessionFactory() throws Exception {

SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean();

sessionFactory.setDataSource(getDataSource());

return sessionFactory.getObject();

}

// 设置utf-8编码

@Override

public void configureMessageConverters(List<HttpMessageConverter<?>> msgConverters) {

StringHttpMessageConverter shmc = new StringHttpMessageConverter(Charset.forName("utf-8"));

msgConverters.add(shmc);

msgConverters.add(new MappingJackson2HttpMessageConverter());

}

/

mvc资源路径配置

/

@Override

public void addResourceHandlers(ResourceHandlerRegistry registry) {

//后面根据路径自己配置

registry.addResourceHandler("/").addResourceLocations("/");

}

完成这2步之后,就实现了无Spring MVC XML自启化。

四、Spring Boot底层源码解析

4.1@EnableAutoConfiguration

@EnableAutoConfiguration借助@Import的支持,网络和注册特定场景干系的bean定义。

例如:(1)@EnableScheduling是通过@Import将Spring调度框架干系的bean定义都加载到IoC容器。

(2)@EnableMBeanExport是通过@Import将JMX干系的bean定义加载到IoC容器。

EnableAutoConfiguration借助@Import的帮助,将所有符合自动配置条件的bean定义加载到IoC容器。

@EnableAutoConfiguration作为一个复合Annotation,其自身定义关键信息如下:

@SuppressWarnings("deprecation")

@Target(ElementType.TYPE)

@Retention(RetentionPolicy.RUNTIME)

@Documented

@Inherited

@AutoConfigurationPackage

@Import(EnableAutoConfigurationImportSelector.class)

public @interface EnableAutoConfiguration {

...

}

4.2SpringFactoriesLoader

SpringFactoriesLoader属于Spring框架私有的一种扩展方案,其紧张功能便是从指定的配置文件META-INF/spring.factories加载配置。

public abstract class SpringFactoriesLoader {

//...

public static <T> List<T> loadFactories(Class<T> factoryClass, ClassLoader classLoader) {

...

}

public static List<String> loadFactoryNames(Class<?> factoryClass, ClassLoader classLoader) {

....

}

}

合营@EnableAutoConfiguration利用的话,它更多是供应一种配置查找的功能支持,即根据@EnableAutoConfiguration的完全类名org.springframework.boot.autoconfigure.EnableAutoConfiguration作为查找的Key,获取对应的一组@Configuration类。
如图

通过META-INF目录下的spring.factories将开启自动配置的类加载初始化。

4.3SpringApplication实行流程

该方法的紧张流程大体可以归纳如下:

这统统都是借助Java SPI机制,读者可自行理解Java SPI机制事理,后面会出文章讲解

(1)根据classpath里面是否存在某个特色类(org.springframework.web.context.ConfigurableWebApplicationContext)来决定是否该当创建一个为Web运用利用的ApplicationContext类型。

(2)利用SpringFactoriesLoader在运用的classpath中查找并加载所有可用的ApplicationContextInitializer。

(3)利用SpringFactoriesLoader在运用的classpath中查找并加载所有可用的ApplicationListener。

(4)推断并设置main方法的定义类。

(5)SpringApplication实例初始化完成并且完成设置后,遍历实行所有通过SpringFactoriesLoader可以查找到并加载的SpringApplicationRunListener。
调用它们的started()方法,开始依次实行程序。

(6)创建并配置当前Spring Boot运用将要利用的Environment

(7)遍历调用所有SpringApplicationRunListener的environmentPrepared()的方法.

(8)根据用户是否明确设置了applicationContextClass类型以及初始化阶段的推断结果,决定该为当前SpringBoot运用创建什么类型的ApplicationContext并创建完成,然后根据条件决定是否添加ShutdownHook,决定是否利用自定义的BeanNameGenerator,决定是否利用自定义的ResourceLoader

(9)ApplicationContext创建好之后,SpringApplication会再次借助Spring-FactoriesLoader,查找并加载classpath中所有可用的ApplicationContext-Initializer,然后遍历调用这些ApplicationContextInitializer的initialize(applicationContext)方法来对已经创建好的ApplicationContext进行进一步的处理。

(10)遍历调用所有SpringApplicationRunListener的contextPrepared()方法。

(11)通过@EnableAutoConfiguration获取的所有配置以及其他形式的IoC容器配置加载到已经准备完毕的ApplicationContext。

(12)遍历调用所有SpringApplicationRunListener的contextLoaded()方法。

(13)调用ApplicationContext的refresh()方法,完成IoC容器可用的末了一道工序。

(14)遍历实行SpringApplicationRunListener的finished()方法

以上都是基于Java SPI装置机制实现的。

五、总结

Spring Boot借助Java Config实现了无XML化配置,借助Java SPI实现自动装置AutoConfiguration,将设计模式利用的无比精妙。
后面会连续深挖Spring Boot实现细节,及微做事技能文章。