从helloworld明白springboot自动配置原理

2018-06-14  本文已影响0人  Lesie_zwc

1、Springboot简介

Spring Boot来简化Spring应用开发,约定大于配置,去繁从简,just run就能创建一个独立的,产品级别的应用背景:J2EE笨重的开发、繁多的配置、低下的开发效率、复杂的部署流程、第三方技术集成难度大。

1-1、解决

“Spring全家桶”时代。
Spring Boot J2EE一站式解决方案
Spring Cloud 分布式整体解决方案

1-2、springboot的优点

1、快速创建独立运行的Spring项目以及与主流框架集成
2、使用嵌入式的Servlet容器,应用无需打成WAR包
3、starters自动依赖与版本控制
4、大量的自动配置,简化开发,也可修改默认值
5、无需配置XML,无代码生成,开箱即用
6、准生产环境的运行时应用监控
7、与云计算的天然集成

2、微服务

微服务:架构风格(服务微化) 一个应用应该是一组小型服务;可以通过HTTP的方式进行互通
单体应用 ALL IN ONE

参考文档

3、SpringBoot Helloworld

3-1、导入spring boot相关的依赖

<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>1.5.10.RELEASE</version>
</parent>
<dependencies>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
</dependencies>

3-2、编写一个主程序,启动spring boot应用

@SpringBootApplication
public class HelloApplication {
    public static void main(String[] args) {
        SpringApplication.run(HelloApplication.class,args);
    }
}

3-3、编写Controller

//@Controller
//@ResponseBody
@RestController //是上面两个注解的简写
public class HelloController {
    @RequestMapping("/hello")
    public String hello(){
        return "hello spring boot";
    }
}

3-4、打包成jar包,运行jar包

<!-- 这个插件,可以将应用打包成一个可执行的jar包;-->
<build>
    <plugins>
        <plugin>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-maven-plugin</artifactId>
        </plugin>
    </plugins>
</build>

4、Helloworld运行原理

4-1、pom文件

父项目

<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>1.5.10.RELEASE</version>
</parent>
  <!--
     spring-boot-starter-parent的父工程是
     spring-boot-dependencies
   -->
   <!--
         spring-boot-dependencies
         来真正的管理所有的第三方依赖和依赖相应的版本
    -->
<parent>
     <groupId>org.springframework.boot</groupId>
     <artifactId>spring-boot-dependencies</artifactId>
     <version>1.5.10.RELEASE</version>
</parent>

以后我们导入依赖默认是不需要写版本;(没有在dependencies里面管理的依赖自然需要声明版本号)

场景启动器依赖

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>

spring-boot-starter:spring-boot场景启动器;帮我们导入了web模块正常运行所依赖的组件;

Spring Boot将所有的功能场景都抽取出来,做成一个个的starters(启动器),只需要在项目里面引入这些starter相关场景的所有依赖都会导入进来。要用什么功能就导入什么场景的启动器

4-2、主程序类、入口类

/**
 *  @SpringBootApplication 来标注一个主程序类,说明这是一个Spring Boot应用
 */
@SpringBootApplication
public class HelloApplication {
    public static void main(String[] args) {
        SpringApplication.run(HelloApplication.class,args);
    }
}

1、@SpringBootApplication:

SpringBoot应用标注在某个类上说明这个类是SpringBoot的主配置类,

SpringBoot就应该运行这个类的main方法来启动SpringBoot应用;

@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(excludeFilters = {
         @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
         @Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {

    ...
}

2、@SpringBootConfiguration

@Configuration
public @interface SpringBootConfiguration {
}
@SpringBootConfiguration:Spring Boot的配置类 标注在某个类上,表示这是一个Spring Boot的配置类
@Configuration:配置类上来标注这个注解 配置类 == 配置文件;配置类也是容器中的一个组件;@Component

3、@EnableAutoConfiguration

@AutoConfigurationPackage
@Import(EnableAutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {
     ...
}

开启自动配置功能; 以前我们需要配置的东西,Spring Boot帮我们自动配置;@EnableAutoConfiguration告诉SpringBoot开启自动配置功能;这样自动配置才能生效;

1、@AutoConfigurationPackage—自动配置包

@Import(AutoConfigurationPackages.Registrar.class)
public @interface AutoConfigurationPackage {
}

导入组件 AutoConfigurationPackages.Registrar.class的作用是将主配置类(@SpringBootApplication标注的类 )所在的包以及所有子包的所有组件扫描到spring容器中

2、@Import(EnableAutoConfigurationImportSelector.class)

作用是导入组件的选择器:
将所有需要导入的组件以全类名的方式返回,这些组件就会被添加到容器中,会给容器中导入非常多的自动配置类(xxxAutoConfiguration),就是给容器中导入这个场景需要的所有组件,并配置好这些组件

有了这些自动配置类,免去我们手动编写配置注入功能组件等工作。

实现原理:在上面选择导入时会调用的一个方法

protected List<String> getCandidateConfigurations(AnnotationMetadata metadata,
         AnnotationAttributes attributes) {
     List<String> configurations = SpringFactoriesLoader.loadFactoryNames(
              getSpringFactoriesLoaderFactoryClass(), getBeanClassLoader());
     Assert.notEmpty(configurations,
              "No auto configuration classes found in META-INF/spring.factories. If you "
                       + "are using a custom packaging, make sure that file is correct.");
     return configurations;
}

上面代码中对应的 loadFactoryNames 方法如下:

public static List<String> loadFactoryNames(Class<?> factoryClass, ClassLoader classLoader) {
     String factoryClassName = factoryClass.getName();
     try {
         Enumeration<URL> urls = (classLoader != null ? classLoader.getResources(FACTORIES_RESOURCE_LOCATION) :
                   ClassLoader.getSystemResources(FACTORIES_RESOURCE_LOCATION));
         List<String> result = new ArrayList<String>();
         while (urls.hasMoreElements()) {
              URL url = urls.nextElement();
              Properties properties = PropertiesLoaderUtils.loadProperties(new UrlResource(url));
              String factoryClassNames = properties.getProperty(factoryClassName);
              result.addAll(Arrays.asList(StringUtils.commaDelimitedListToStringArray(factoryClassNames)));
         }
         return result;
     }
     catch (IOException ex) {
         throw new IllegalArgumentException("Unable to load [" + factoryClass.getName() +
                  "] factories from location [" + FACTORIES_RESOURCE_LOCATION + "]", ex);
     }
}

public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";

从上面的源码可以看出,自动配置的核心是: Spring Boot在启动的时候从类路径下的META-INF/spring.factories中获取EnableAutoConfiguration指定的值,将这些值作为自动配置类导入到容器中,自动配置类就生效,帮我们进行自动配置工作;以前我们需要自己配置的东西,自动配置类都帮我们;

上一篇下一篇

猜你喜欢

热点阅读