Springboot的理解和疑惑

前言:大家都知道springboot的好处是,简化配置,一键启动,面试也总是问自动装配的原理,我们只有深刻理解装配原理才能以不变应万变。

目录

问题1:springboot 自动配置是如何 知道依赖的类存在不存在?

问题2: springboot-start-XXXX下META-INF的文件作用


问题1:springboot 自动配置是如何 知道依赖的类存在不存在?

在Spring Boot中,自动配置是通过类路径上的META-INF/spring.factories文件实现的。这个文件中列出了所有自动配置类的全限定类名。Spring Boot在启动时会扫描这些自动配置类,然后根据需要将它们应用到应用程序上下文中。

当您在项目中添加一个新的依赖时,例如在pom.xml文件中添加一个新的库,Spring Boot会扫描该库的类路径以查找META-INF/spring.factories文件。如果该文件存在,它将被读取并将其中列出的自动配置类加载到应用程序上下文中。

如果自动配置类依赖于其他类或库,但这些类或库不存在,或者没有正确配置,Spring Boot会在应用程序启动时抛出异常。因此,要确保自动配置类正常工作,必须正确配置依赖项并将它们添加到类路径中。

总的来说,Spring Boot的自动配置机制是基于类路径上的META-INF/spring.factories文件的。这个文件中列出了所有自动配置类的全限定类名。当您添加新的依赖时,Spring Boot会自动扫描该依赖的类路径以查找该文件,并将其中列出的自动配置类应用到应用程序上下文中。

扩展:所以,我们也了解到:如果发现jar包中的类是报错的,如:

那这个自动装配是肯定不会 自动装配的。 

问题2: springboot-start-XXXX下META-INF的文件作用

spring-autoconfigure-metadata.properties文件是Spring Boot自动配置元数据文件,用于描述Spring Boot自动配置类的元数据信息。这个文件包含一组键值对,每个键值对表示一个自动配置项的元数据信息,例如属性名称、属性类型、默认值、描述等等。

Spring Boot使用这个文件来帮助IDE(例如Eclipse、IntelliJ IDEA等)和其他工具(例如Spring Boot Actuator)了解自动配置类的信息

如:

org.springframework.boot.autoconfigure.condition.ConditionalOnProperty=\
  name,spring.datasource.url,\
  matchIfMissing=false
  prefix=spring.datasource

spring-configuration-metadata.json文件是Spring Boot配置元数据文件,用于描述应用程序配置属性的元数据信息。这个文件包含了一组JSON格式的键值对,每个键值对表示一个配置属性的元数据信息,例如属性名称、属性类型、默认值、描述等等。

Spring Boot使用这个文件来帮助IDE(例如Eclipse、IntelliJ IDEA等)和其他工具(例如Spring Boot Actuator)了解自动配置类的信息

如:

{
  "name": "spring.datasource.url",
  "type": "java.lang.String",
  "description": "URL of the database.",
  "defaultValue": "jdbc:h2:mem:testdb"
}

spring.factories文件是Spring框架的一个标准配置文件,它允许第三方开发者向Spring框架注册自己的实现类或者工厂类,以扩展或替换Spring框架的默认行为。这个文件位于类路径的META-INF目录下,它是一个标准的Java属性文件,其中每行包含了一个键值对,键表示要注册的类型或接口,值表示该类型或接口的实现类或工厂类。

如:

org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
  org.springframework.boot.autoconfigure.web.WebMvcAutoConfiguration,\
  org.springframework.boot.autoconfigure.web.HttpMessageConvertersAutoConfiguration,\
  org.springframework.boot.autoconfigure.jackson.JacksonAutoConfiguration

 SpringBoot的启动原理:

手写SpringBoot源码: 

启动类:

@ZhouyuSpringBootApplication
public class MyApplication {

    public static void main(String[] args) {
        ZhouyuSpringApplication.run(MyApplication.class);
    }
}


------------以下是启动类注解--------
/*

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Configuration
@ComponentScan //只能扫描当前注解标注类的 兄弟及下属目录 扫描到的且标注注解的 才放入容器
@Import(ZhouyuImportSelect.class) // @import的作用是将ZhouyuImportSelect.class作为一个bean放入容器
public @interface ZhouyuSpringBootApplication {
}

*/

1、创建容器(创建springBoot容器、开启Tomcat服务器)

public class ZhouyuSpringApplication {

    public static void run(Class clazz){
        //创建一个spring容器  创建的方式有很多种,如果是加载xml的话,就是ClassPathXmlApplicationContext,如果是加载注解的话,就是AnnotationConfigApplicationContext
        AnnotationConfigWebApplicationContext applicationContext = new AnnotationConfigWebApplicationContext();
        applicationContext.register(clazz);//向容器中注册一个启动类
        applicationContext.refresh();
        //从sspring容器中开启TomcatService容器
        WebServer webServer = getWebServer(applicationContext);
        webServer.start();

    }

    public static WebServer getWebServer(ApplicationContext applicationContext){
        Map<String, WebServer> beansOfType = applicationContext.getBeansOfType(WebServer.class);
        if (beansOfType.isEmpty()) {
            throw new NullPointerException();
        }
        if (beansOfType.size() > 1) {//如果拿到多个同类型的Bean就抛异常
            throw new IllegalStateException();
        }
        return beansOfType.values().stream().findFirst().get();
    }
}

2、向容器放入需要Bean

//这个类需要注意 有没有被 SpringBoot扫描到?这时候就需要在 启动类头上只扫描了其兄弟目录
@Configuration
public class WebServerAutoConfiguration implements AutoConfiguration {

    @Bean
    @ConditionalOnClass(name = "org.apache.catalina.startup.Tomcat")  
//如果maven中排除Tomcat依赖 这个就为false 此刻该bean就注入不成功了,其实现是通过类加载器获取,获取不到表示系统中没有这个类
    public TomcatWebServer tomcatWebServer(){
        return new TomcatWebServer();
    }

    @Bean
    @ConditionalOnClass(name = "org.eclipse.jetty.server.Server")
    public JettyWebServer jettyWebServer(){
        return new JettyWebServer();

    }
}

 通过配置类向容器中注入 TomcatService和JettyService,但上面代码写的,发现两个webServers时则抛异常,所以我们对Bean的注入加入@Conditional(XX.class)(其内部)通过类加载器判断有没有加载到类。下图是实现判定是否存在该class的原理:

 Maven知识扩展:

<option>true<option>标签:

猜你喜欢

转载自blog.csdn.net/u013372493/article/details/129560787