SpringBoot底层原理

SpringBoot底层原理

一 配置优先级

1.配置方式

Springboot中支持三种配置方式,分别为:

  • application.properties
  • application.yml
  • application.yaml

2.配置优先级

当存在多份配置文件时,配置文件会按照它们的优先级生效。

优先级从高到底分别为:application.peoperties>application.yml>application.yaml

目前 application.yml 是最主流的方式

3.其他配置方式

Springboot除了以上常见的三种配置方式之外,还支持Java系统属性配置和命令行参数配置。

1.Java系统属性配置示例

# 在 java 命令后使用 —D 命令,然后书写需要配置的属性即可
# 示例中配置了项目的运行端口为8081,即server.port=8081
java -Dserver.port=8081 -jar [packageName].jar

2.命令行参数配置示例

# 在jar包名称之后,使用双横杠(--),后面紧跟配置的参数即可
java -jar [packageName].jar --server.port=8082

二 Bean管理

1.获取bean对象

默认情况下,Springboot项目在启动时会自动创建bean,并且将这些bean都存放在IOC容器中。

如果想手动获取这些bean,则可以通过以下几种示例。

首先需要注入ApplicationContext对象。

Spring框架中,ApplicationContext是一个接口,代表了Spring容器,它负责管理Spring应用程序中所有的bean,同时提供了一些方法来获取Bean,注册Bean,是整个Spring应用的核心。

默认情况下,一个Bean的名称是它的类名名称,然后将首字母小写。

例如 DeptController,它在IOC容器中的默认名称为 deptController

@Autowired
private ApplicationContext applicationContext;
@Test
void test1(){
    //1.根据bean的名称来获取bean对象
    EmpServiceImpl empServiceImpl1= (EmpServiceImpl) applicationContext.getBean("empServiceImpl");
    System.out.println(empServiceImpl1);
    
    //2.根据bean的类型来获取bean对象
    EmpServiceImpl empServiceImpl2= applicationContext.getBean(EmpServiceImpl.class);
    System.out.println(empServiceImpl2);
    
    //3.根据bean的类型和名称来获取bean对象
    // 以下方法和示例,在bean的默认名称被修改且有多个同类型的bean时,尤为有用。
    EmpServiceImpl empServiceImpl3= applicationContext.getBean("empServiceImpl",EmpServiceImpl.class);
    System.out.println(empServiceImpl3);
}

运行以上示例,可以看出,bean在IOC容器中,默认是单例存在的。
在这里插入图片描述如果想要实现每次使用时都是一个新的bean,则需要通过bean的作用域来进行配置。

2.bean的作用域

在Spring中,bean支持五种作用域,后三种在web环境下才能生效。

在这里插入图片描述

默认情况下,Bean对象在项目启动时就会默认实例化。如果不希望在项目启动时就初始化,可以使用@Lazy注解,让Bean对象延迟初始化,直到第一次使用该Bean时才会进行初始化。

@Service
@Lazy
public class EmpServiceImpl extends ServiceImpl<EmpMapper, Emp> implements EmpService{
    //code...
}

设置Bean的作用域,则需要通过@Scope注解来实现。

以下示例中,我们将Bean的作用域设置为 prototype,即每次使用该Bean时都会创建新对象。

@Service
@Scope("prototype")
public class EmpServiceImpl extends ServiceImpl<EmpMapper, Emp> implements EmpService{
    //code...
}

现在再次运行前边获取Bean对象的代码示例,可以发现,三次获取到的Bean对象,已经不是同一个。
在这里插入图片描述

3.声明第三方bean

如果要申明的Bean来自第三方,是无法通过@Component及衍生注解来申明的,这个时候就需要使用@Bean注解。

以 SAXReader 类为例,创建一个返回值为SAXReader对象的方法,方法名称就是以后被ICO管理的Bean对象名称。

//将方法的返回值交给IOC容器管理,称为IOC容器的Bean对象
@Bean
public SAXReader saxReader(){
    return new SAXReader();
}

在以后需要用到 SAXReader 对象的时候,直接注入即可,不用去实例化。

@Autowired
private SAXReader saxReader;
@Test
void test2() throws DocumentException {
    Document document = saxReader.read("xxx");
}

注意:一般情况下,我们通常会将所有需要申明的第三方bean对象统一放在一个配置类中,这样更加方便维护。

@Configuration
public class BeanAutoConfig {
    // 可以通过@Bean注解的 name /value 属性来定义bean的名称
    // 默认情况下,bean的名称就是方法名
    @Bean
    public RestTemplate restTemplate(){
        return new RestTemplate();
    }
    
}

在声明第三方bean对象时,如果需要进行依赖注入,则只需要指定方法形参即可。Spring会根据类型进行自动装配。

// EmpServiceImpl对象是要注入的bean对象
@Bean
public RestTemplate restTemplate(EmpServiceImpl empServiceImpl){
    empServiceImpl.[xxx];
    return new RestTemplate();
}

三 Springboot原理

1.Springboot起步依赖

Springboot整合了以前web开发需要用的一些依赖项,目前使用Springboot开发web项目,只需要引入 spring-boot-start-web 依赖即可。
在这里插入图片描述
归根结底,SPringboot起步依赖的原理就是maven的依赖传递

2.自动配置

当Springboot项目启动后,Springboot中的一些配置类,bean对象就会自动存入到IOC容器中,不需要我们手动去申明。从而简化了开发,省去了繁琐的配置。

3.管理第三方包中的Bean

1.配置实现方式1:@ComponentScan 组件扫描

在启动类上使用 @ComponentScan注解,重新配置包扫描路径。@ComponentScanbasePackages参数支持数据格式,当有多个第三方包时,可使用数组形式申明。

@ComponentScan(basePackages = "xxx1")

@ComponentScan(basePackages = {
   "xxx1",
   "xxx2"
})

注意:使用@ComponentScan申明时,当前Springboot项目中的包路径也必须包含在内,否则当前项目中的bean将会被无法识别。

缺点:当项目较大时,引入大量的第三方依赖,此时启动类将会显得臃肿。

2.配置实现方式2:@Import 组件导入

使用@Import导入的类会被Spring加载到容器中。可以导入普通类,配置类,以及 ImportSelector 接口的实现类,支持数组。

@Import({xx1.class,xx2.class})

实现ImportSelector 接口,这里最核心的就是 selectImports方法,它返回了需要创建Bean对象的全部类。

public class MyImportSelector implements ImportSelector {
    @Override
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
        //此处的数组内容为需要导入的Bean对象的全类名,有多少写多少
        return new String[]{"springboot.demo.GoodStudent"};
    }
}

然后在启动类上使用@Import注解即可。

@Import({MyImportSelector.class})

单纯使用 @Import接口声明第三方Bean,缺点很明显。

3.配置实现方式3:第三方依赖提供注解

实际项目中,具体第三方依赖需要导入哪些Bean,只有第三方依赖自己知道。所以可以由第三方依赖提供注解,然后在项目中引入即可。

此类型注解一般均由 Enable 开头,原理是在自定义注解中封装 @Import 注解。

示例:

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Import({
        MybatisPlusConfig.class,
        CorsConfig.class,
        SpringWebConfig.class,
        BeanAutoConfig.class,
        RedisConfig.class
})
public @interface EnableShawnConfig {
    
}

在第三方依赖包中定义一个EnableShawnConfig注解,然后使用 @Import将需要配置的Bean对象都引入进来。最后再在项目中使用该注解即可。

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

4.自动配置原理分析

Springboot核心注解:@SpringBootApplication

@SpringBootApplication下的重要注解:

  • @SpringBootConfiguration:申明当前注解也是一个配置类,因为@SpringBootConfiguration中也申明了@Configuration注解。所以可以直接在启动类中申明第三方的bean对象。
  • @EnableAutoConfiguration:Springboot自动配置的核心注解,它声明了@Import(AutoConfigurationImportSelector.class)AutoConfigurationImportSelectorImportSelector接口的实现类,实现了 selectImports 方法selectImports 方法返回了需要创建Bean对象的全部信息。
  • @ComponentScan:组件扫描,默认扫描当前引导类及其所在的子包。

自动配置最核心的注解就是@EnableAutoConfiguration,由源码可知,selectImports 方法会读取一个固定目录下后缀名为.imports的文件。

protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
    List<String> configurations = ImportCandidates.load(AutoConfiguration.class, getBeanClassLoader())
            .getCandidates();
    Assert.notEmpty(configurations,
            "No auto configuration classes found in "
                    + "META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports. If you "
                    + "are using a custom packaging, make sure that file is correct.");
    return configurations;
}

springboot3.x中,META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports文件中,配置了需要创建Bean对象的全类名

在Springboot2.x中,是通过两个关键文件来读取配置好的全类名的。
在这里插入图片描述
spring.factories 是早起Springboot版本中自动配置的文件,在后续版本中已经逐渐不再使用。

5.@Conditinal注解

@Conditinal注解的作用:按照一定的条件判断,在满足条件后才会注册对应的Bean对象到IOC容器中

它可以作用在方法上。

@Conditinal本身是一个父级注解,它衍生除了很多子级注解

  • @ConditionalOnClass:判断环境中是否存在字节码文件,有则注册bean对象到IOC容器
  • @ConditionalOnMissingBean:判断环境中有没有对应的Bean(根据类型和名称),没有则注册Bean对象到IOC容器
  • @ConditionalOnProperty:判断配置文件中是否有对应属性和值,有则注册bean对象到IOC容器

相关文章

  • SQL实现模糊查询的四种方法总结

    SQL实现模糊查询的四种方法总结

  • Spring两大核心思想:IOC和AOP

    1.先自定义一个注解@Target(ElementType.METHOD) //作用范围 此处是方法@Retention(RetentionPolicy.RUNTIME) //生命周期 此处是运行时2.在切点表达式中加入该注解@Aspect@Component@Slf4j@Around(&quot;@annotation(com.example.demo.aspect.TestAnnotation)&quot;) //此处加自定义注解log.info(&quot;around继续&quot;);try {

  • SpringCloud-搭建Nacos配置中心

    本文详细介绍了如何在Spring Cloud项目中使用Nacos实现配置管理。首先,通过简洁的步骤指导了Nacos的安装和配置。然后,通过在项目中引入Nacos的依赖和配置,实现了与Nacos配置中心的连接。在Nacos控制台上演示了如何新建配置,并通过Spring Cloud项目实现了动态读取配置的操作。这种灵活的配置管理方案为微服务架构提供了高度可维护性和实时性的优势,使得项目能够在运行时动态调整配置,而无需重启服务

  • C#使用重载方法实现不同类型数据的计算

    为了避免异常,可以先使用Decimal.Parse(string)方法将字符串转换为小数,然后再使用Convert.ToInt32(decimal)方法将小数转换为整数。如果一个类中存在两个以上的同名方法,并且方法的参数类型、个数或者顺序不同,当调用这样的方法时,编译器会根据传入的参数自动进行判断,决定调用哪个方法。例如,字符串是&quot;123.456&quot;,包含非数字字符&quot;.&quot;。重载方法就是方法名称相同,但是每个方法中参数的数据类型、个数或顺序不同的方法。如果字符串包含非数字字符,例如小数点,该方法将引发异常。

  • 小程序 自定义组件和生命周期

    类似vue或者react中的自定义组件⼩程序允许我们使⽤⾃定义组件的⽅式来构建⻚⾯。类似于页面,一个自定义组件由 json wxml wxss js 4个文件组成可以在微信开发者⼯具中快速创建组件的⽂件结构在⽂件夹内 components/myHeader ,创建组件 名为 myHeader⾸先要在⻚⾯的 json ⽂件中进⾏引⽤声明。还要提供对应的组件名和组件路径index.wxml// 引用声明// 要使用的组件的名称 // 组件的路径&lt; view &gt; _微信小程序 自定义组件生命周期

  • Java把列表数据导出为PDF文件,同时加上PDF水印

    可以看到字体文件在jar目录下面是有的,但是发现classes后面多了个叹号。这是引入外部字体方式不对,后改用问题2参考文章的第三种写法就没问题了。网上都是说jar包的版本不对,导致的字体兼容性问题。换了jar包版本发现没效果,后来索性直接把字体下载到本地直接引入。字体文件资源自己百度,直接搜。

  • HarmonyOS 鸿蒙应用开发( 六、实现自定义弹窗CustomDialog)

    自定义弹窗(CustomDialog)可用于广告、中奖、警告、软件更新等与用户交互响应操作。开发者可以通过CustomDialogController类显示自定义弹窗。具体用法请参考自定义弹窗。

  • java获取双异步返回值时,如何保证主线程不阻塞?

    CompletableFuture的异步执行通过ForkJoinPool实现,ForkJoinPool在于可以充分利用多核CPU的优势,把一个任务拆分成多个小任务,把多个小任务放到多个CPU上并行执行,当多个小任务执行完毕后,再将其执行结果合并起来。

  • Vue3前端开发,如何获取组件内dom对象以及子组件的属性和方法

    传统的vue2里面,我们访问dom时的代码,还是的借助于this对象的【this.$refs.userName】。毕竟,父子组件之间,各司其职。下面展示的是,借助于ref来访问子组件的实例对象。(声明:默认情况下,子组件内部的属性和方法,不会主动对外暴漏的。Vue3前端开发,借助Ref来获取组件内dom对象,借助defineExpose编译宏可以获取到子组件的属性和方法。如图,确实是可以拿到子组件的属性和方法了,如果不使用编译宏,是访问不到的。这个是子组件里面的内容,我们定义了一个常量,一个方法。

  • 掌握Spring MVC拦截器整合技巧,实现灵活的请求处理与权限控制!

    在这个方法中可以通过返回值来决定是否要进行放行,我们可以把业务逻辑放在该方法中,如果满足业务则返回true放行,不满足则返回false拦截。方法,如果返回true,则代表放行,会执行原始Controller类中要请求的方法,如果返回false,则代表拦截,后面的就不会再执行了。(7)如果满足规则,则进行处理,找到其对应的controller类中的方法进行执行,完成后返回结果。配置多个后,执行顺序是什么?(5)在找到具体的方法之前,我们可以去配置过滤器(可以配置多个),按照顺序进行执行。

  • Java编程模型:VO,BO,PO,DO,DTO

    Java编程模型中的VO,BO,PO,DO,DTO提供了一种结构化和组织代码的方法。通过合理运用这些概念,可以使代码更具可读性、可维护性和可扩展性。在实际项目中,根据需求和架构设计,合理选择和运用这些概念将有助于构建清晰、高效的Java应用程序。