@SpringBootApplication注解分析

本文深入解析Spring Boot启动注解@SpringBootApplication的功能,包括@ComponentScan、@SpringBootConfiguration和@EnableAutoConfiguration的作用。详细阐述如何使用@SpringBootApplication注解替代@ComponentScan注解实现程序运行,以及@SpringBootApplication参数详解,包括exclude、excludeName、scanBasePackages和scanBasePackageClasses的使用场景。并通过实例演示配置类、实体类及组件类如何被Spring容器管理。

首先我们分析的就是入口类Application的启动注解@SpringBootApplication,进入源码:

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(excludeFilters = {
        @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
        @Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {
....
}

发现@SpringBootApplication是一个复合注解,包括@ComponentScan,和@SpringBootConfiguration@EnableAutoConfiguration

  • @SpringBootConfiguration继承自@Configuration,二者功能也一致,标注当前类是配置类,并会将当前类内声明的一个或多个以@Bean注解标记的方法的实例纳入到srping容器中,并且实例名就是方法名。
  • @EnableAutoConfiguration的作用启动自动的配置,@EnableAutoConfiguration注解的意思就是Springboot根据你添加的jar包来配置你项目的默认配置,比如根据spring-boot-starter-web ,来判断你的项目是否需要添加了webmvctomcat,就会自动的帮你配置web项目中所需要的默认配置。在下面博客会具体分析这个注解,快速入门的demo实际没有用到该注解。
  • @ComponentScan,扫描当前包及其子包下被@Component@Controller@Service@Repository注解标记的类并纳入到spring容器中进行管理。是以前的<context:component-scan>(以前使用在xml中使用的标签,用来扫描包配置的平行支持)。所以本demo中的User为何会被spring容器管理。

根据上面的理解,上面的入口类Application,我们可以使用:

package com.zhihao.miao;
import com.zhihao.miao.bean.User;
import org.springframework.boot.SpringApplication;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import java.util.Map;

@ComponentScan
public class Application {

    @Bean
    public Runnable createRunnable(){
        return () -> System.out.println("spring boot is running");
    }

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(Application.class,args);
        context.getBean(Runnable.class).run();
        System.out.println(context.getBean(User.class));
        Map map = (Map) context.getBean("createMap");
        int age = (int) map.get("age");
        System.out.println("age=="+age);

    }
}

使用@ComponentScan注解代替@SpringBootApplication注解,也可以正常运行程序。原因是@SpringBootApplication中包含@ComponentScan,并且springboot会将入口类看作是一个@SpringBootConfiguration标记的配置类,所以定义在入口类Application中的Runnable也可以纳入到容器管理。

SpringBootApplication参数详解


图片.png
  • Class<?>[] exclude() default {}:
    根据class来排除,排除特定的类加入spring容器,传入参数value类型是class类型。
  • String[] excludeName() default {}:
    根据class name来排除,排除特定的类加入spring容器,传入参数value类型是class的全类名字符串数组。
  • String[] scanBasePackages() default {}:
    指定扫描包,参数是包名的字符串数组。
  • Class<?>[] scanBasePackageClasses() default {}:
    扫描特定的包,参数类似是Class类型数组。

看一个demo学会使用这些参数配置
在包下com.zhihao.miao.springboot定义一个启动应用类(加上@SpringBootApplication注解)

package com.zhihao.miao.springboot;


import com.zhihao.miao.beans.Cat;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;

@SpringBootApplication
public class Application {

    public static void main(String[] args) {
        ConfigurableApplicationContext context =SpringApplication.run(Application.class,args);
        Cat cat = context.getBean(Cat.class);
        System.out.println(cat);
    }
}

在com.zhihao.miao.beans包下定义一个实体类,并且想将其纳入到spring容器中,

public class Cat {
}
package com.zhihao.miao.beans;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class MyConfig {

    @Bean
    public Cat cat(){
        return new Cat();
    }

}

启动启动类,打印结果如下:


 

说明Cat类并没有纳入到spring容器中,这个结果也如我们所想,因为@SpringBootApplication只会扫描@SpringBootApplication注解标记类包下及其子包的类(特定注解标记,比如说@Controller,@Service,@Component,@Configuration和@Bean注解等等)纳入到spring容器,很显然MyConfig不在@SpringBootApplication注解标记类相同包下及其子包的类,所以需要我们去配置一下扫包路径。

修改启动类,@SpringBootApplication(scanBasePackages = "com.zhihao.miao"),指定扫描路径:

package com.zhihao.miao.springboot;

import com.zhihao.miao.beans.Cat;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;

@SpringBootApplication(scanBasePackages = "com.zhihao.miao")
public class Application {

    public static void main(String[] args) {
        ConfigurableApplicationContext context =SpringApplication.run(Application.class,args);
        Cat cat = context.getBean(Cat.class);
        System.out.println(cat);
    }
}

启动并打印:


 

当然使用@SpringBootApplication(scanBasePackageClasses = MyConfig.class),指定scanBasePackageClasses参数的value值是你需要扫描的类也可以,结果一样,不过如果多个配置类不在当前包及其子包下,则需要指定多个。

再看一个列子,
在上面的列子的相同包下(com.zhihao.miao.springboot)配置了People,并将其纳入到spring容器中(@Component),我们知道@SpringBootApplication注解会扫描当前包及其子包,所以People类会纳入到spring容器中去,我们需要将其排除在spring容器中,如何操作?
可以使用@SpringBootApplication的另外二个参数(exclude或excludeName)

package com.zhihao.miao.springboot;

import org.springframework.stereotype.Component;

@Component
public class People {
}

启动类,

package com.zhihao.miao.springboot;


import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;

@SpringBootApplication
public class Application {

    public static void main(String[] args) {
        ConfigurableApplicationContext context =SpringApplication.run(Application.class,args);
        People people = context.getBean(People.class);
        System.out.println(people);
    }
}

启动并打印结果:


 

然后修改@SpringBootApplication配置,

package com.zhihao.miao.springboot;


import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;

@SpringBootApplication(exclude = People.class)
public class Application {

    public static void main(String[] args) {
        ConfigurableApplicationContext context =SpringApplication.run(Application.class,args);
        People people = context.getBean(People.class);
        System.out.println(people);
    }
}

很明显启动报错。使用@excludeName注解也可以。如下,
@SpringBootApplication(excludeName = {"com.zhihao.miao.springboot.People"})

参考文档:
Springboot1.5.4官方文档



作者:二月_春风
链接:http://www.jianshu.com/p/4e1cab2d8431
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
<think>我们正在讨论@SpringBootApplication注解。根据引用内容,该注解是SpringBoot的核心注解,它简化了SpringBoot应用的配置过程。下面我将详细解释该注解的功能、组成和使用方法。###功能@SpringBootApplication注解的主要功能是简化SpringBoot应用的配置。它通过组合三个注解来实现:1.**配置简化**:替代了传统的多个配置类。2.**自动装配**:根据项目依赖自动配置Spring应用。3.**组件扫描**:自动扫描当前包及其子包中的组件(如@Service,@Controller等)。###组成从引用[1][2][3][4]中可以看出,@SpringBootApplication是一个组合注解,它包含三个核心注解:1.**@SpringBootConfiguration**:-表明该类是一个配置类,允许在上下文中注册额外的Bean或导入其他配置类。-它本身是@Configuration的派生注解,因此具有相同的功能[^4]。2.**@EnableAutoConfiguration**:-启用SpringBoot的自动配置机制,根据类路径中的jar包依赖自动配置Bean。-例如:如果类路径中有`spring-boot-starter-web`,则自动配置Tomcat和SpringMVC[^4]。3.**@ComponentScan**:-自动扫描当前包及其子包下的组件(如@Component,@Service,@Repository,@Controller等),并将它们注册为SpringBean。-默认扫描启动类所在包及其所有子包[^3]。###使用方法####1.基本用法将@SpringBootApplication注解标注在**主启动类**上:```java@SpringBootApplication//核心注解publicclassApplication{publicstaticvoidmain(String[]args){SpringApplication.run(Application.class,args);//启动SpringBoot}}```####2.常用属性@SpringBootApplication提供了以下常用属性(引用[1]):-**scanBasePackages**:自定义组件扫描的包路径。```java@SpringBootApplication(scanBasePackages={"com.example.service","com.example.controller"})```-**exclude**:显式排除特定的自动配置类。```java@SpringBootApplication(exclude={DataSourceAutoConfiguration.class})```-**excludeName**:通过类名排除自动配置类(字符串形式)。```java@SpringBootApplication(excludeName={"org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration"})```####3.高级场景-**多配置类支持**:如果需拆分配置,可用`@Import`导入其他配置类:```java@SpringBootApplication@Import({SecurityConfig.class,SwaggerConfig.class})publicclassApplication{...}```-**禁用组件扫描**:若需完全手动控制Bean注册,可移除`@ComponentScan`:```java@SpringBootConfiguration@EnableAutoConfigurationpublicclassApplication{...}//此时需手动注册所有Bean```###自动装配原理(引用[4])1.**条件装配**:`@EnableAutoConfiguration`利用`@Conditional`系列注解(如`@ConditionalOnClass`)按需加载配置。2.**加载路径**:从`META-INF/spring.factories`文件中读取自动配置类列表。3.**优先级**:用户自定义配置优先于自动配置(如定义自己的`DataSource`Bean会覆盖自动配置)。---###§相关问题§1.如何覆盖SpringBoot的自动配置?2.`@SpringBootApplication`的扫描范围如何影响Bean的注册?3.在SpringBoot中如何排除特定的自动配置类?4.`@SpringBootConfiguration`和`@Configuration`有什么区别?5.如何自定义`spring.factories`文件实现自己的自动配置?</think>### @SpringBootApplication 注解详解 #### 一、核心功能 @SpringBootApplication 是 Spring Boot 的核心注解,通过组合三个关键注解实现三大功能: 1. **配置简化**:替代传统 Spring 的 XML 配置 2. **自动装配**:根据依赖自动配置组件(如数据库连接、Web 服务器等) 3. **组件扫描**:自动发现并注册 Bean(如 `@Controller`, `@Service` 等)[^3][^4] #### 二、注解组成 从源码角度分析其结构(引用[1][2][4]): ```java @SpringBootConfiguration @EnableAutoConfiguration @ComponentScan public @interface SpringBootApplication { ... } ``` | 子注解 | 功能说明 | 示例场景 | |-------------------|--------------------------------------------------------------------------|------------------------------| | `@SpringBootConfiguration` | 标识主配置类(实质是 `@Configuration` 的特化) | 定义自定义 Bean 的配置类 | | `@EnableAutoConfiguration` | 启用自动装配机制,根据类路径依赖自动配置组件 | 自动配置数据源、Web MVC 等 | | `@ComponentScan` | 扫描当前包及子包的 Spring 组件(`@Component`, `@Service` 等) | 自动注册 Controller 和 Service | #### 三、关键属性配置 通过属性可定制化行为(引用[1][4]): ```java @SpringBootApplication( scanBasePackages = "com.example", // 自定义扫描路径 exclude = {DataSourceAutoConfiguration.class}, // 排除特定自动配置 excludeName = "org.example.CustomConfig" // 按类名排除配置 ) ``` #### 四、使用方法 1. **基础用法**(90% 场景适用): ```java @SpringBootApplication public class MyApp { public static void main(String[] args) { SpringApplication.run(MyApp.class, args); // 启动Spring Boot } } ``` - 默认扫描**启动类所在包及其子包** - 自动配置 Tomcat/Jetty、Spring MVC 等组件 2. **多模块项目配置**: ```java @SpringBootApplication(scanBasePackages = { "com.module.a", "com.module.b" }) ``` 3. **排除自动配置**(如禁用数据库): ```java @SpringBootApplication(exclude = DataSourceAutoConfiguration.class) ``` #### 五、工作原理 1. **启动流程**: ```mermaid graph TD A[SpringApplication.run] --> B[加载@SpringBootApplication] B --> C[@ComponentScan 注册Bean] B --> D[@EnableAutoConfiguration 加载自动配置] D --> E[读取META-INF/spring.factories] E --> F[按条件装配Bean] ``` 2. **自动装配机制**: - 检查类路径依赖(如存在 `spring-boot-starter-web` → 自动配置 Tomcat) - 通过 `@Conditional` 系列注解实现条件化装配(如 `@ConditionalOnClass`) - 配置优先级:用户自定义 Bean > 自动配置 Bean[^4] #### 六、最佳实践 1. **位置要求**:必须放在**主启动类**上(根包或接近根包位置) 2. **自定义配置**:优先使用属性配置而非覆盖自动配置类 3. **调试技巧**:启动时添加 `--debug` 参数查看自动配置报告 ``` Positive matches: WebMvcAutoConfiguration matched DataSourceAutoConfiguration matched ``` --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值