文章目录
- Spring Boot拓展注解@SpringBootApplication和@Configuration
- 0x01_@SpringBootApplication
- @SpringBootConfiguration
- @EnableAutoConfiguration
- @ComponentScan
- 0x02_@Configuration
- Full 全模式,Lite 轻量级模式
Spring Boot拓展注解@SpringBootApplication和@Configuration
0x01_@SpringBootApplication
这个注解在分析Spring boot启动原理时,看过源码分析过,在这个注解的源码中,最重要的是其上的几个注解:
如果在项目的启动类中,不用@SpringBootApplication
注解,而用上面的3个注解,一样可以启动:
注意:
@ComponentScan
要指定扫描的包。
package com.bones;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.SpringBootConfiguration;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.ComponentScan;
//@SpringBootApplication
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan("com.bones")
public class SpringbootannoApplication {
public static void main(String[] args) {
SpringApplication.run(SpringbootannoApplication.class, args);
}
}
关于@SpringBootApplication
注解里面的3个子注解,下面详细分析一下:
@SpringBootConfiguration
为什么@SpringBootApplication
注解里没有包含@Configuration
,实际上是在@SpringBootConfiguration
里面
@SpringBootConfiguration
注解源码:
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Configuration
@Indexed
public @interface SpringBootConfiguration {
@AliasFor(annotation = Configuration.class)
boolean proxyBeanMethods() default true;
}
@SpringBootConfiguration继承自@Configuration,二者功能也一致,标注当前类是配置类, 并会将当前类内声明的一个或多个以@Bean注解标记的方法的实例纳入到spring容器中,并且实例名就是方法名。
只不过@SpringBootConfiguration是springboot的注解,而@Configuration是spring的注解
从上述源码可以看出,@SpringBootConfiguration注解内部有一个核心注解@Configuration,该注解是Spring框架提供的,表示当前类为一个配置类(XML配置文件的注解表现形式),并可以被组件扫描器扫描。由此可见,@SpringBootConfiguration注解的作用与@Configuration注解相同,都是标识一个可以被组件扫描器扫描的配置类,只不过@SpringBootConfiguration是被Spring Boot进行了重新封装命名而已。
@EnableAutoConfiguration
这个注解的作用:
帮助SpringBoot应用将所有符合条件的@Configuration
配置都加载到当前SpringBoot,并创建对应配置类的Bean,并把该Bean实体交给IoC容器进行管理。
@EnableAutoConfiguration
的源码
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import(AutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {
String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";
Class<?>[] exclude() default {};
String[] excludeName() default {};
}
@EnableAutoConfiguration 注解也是一个组合注解。其中:
- exclude():排除特定的自动配置类,使它们永远不会被应用.
- excludeName():排除特定的自动配置类名,使它们永远不会被应用.
通常情况下,我们不需要显示使用 @EnableAutoConfiguration
注解。因为在 @SpringBootApplication
注解上面声明了 @EnableAutoConfiguration
注解。
@EnableAutoConfiguration
实现的关键在于引入了AutoConfigurationImportSelector
,其核心逻辑为selectImports
方法,借助AutoConfigurationImportSelector
,它可以帮助SpringBoot应用将所有符合条件的@Configuration
配置都加载到当前SpringBoot创建并使用的IoC容器。
当springboot扫描到@EnableAutoConfiguration
注解时则会将spring-boot-autoconfigure.jar/META-INF/spring.factories
文件(上面的截图)中org.springframework.boot.autoconfigure.EnableAutoConfiguration
对应的value里的所有xxxConfiguration类加载到IOC容器中。spring.factories文件里每一个xxxAutoConfiguration文件一般都会有下面的条件注解:
@ConditionalOnClass : classpath中存在该类时起效
@ConditionalOnMissingClass : classpath中不存在该类时起效
@ConditionalOnBean : DI容器中存在该类型Bean时起效
@ConditionalOnMissingBean : DI容器中不存在该类型Bean时起效
@ConditionalOnSingleCandidate : DI容器中该类型Bean只有一个或@Primary的只有一个时起效
@ConditionalOnExpression : SpEL表达式结果为true时
@ConditionalOnProperty : 参数设置或者值一致时起效
@ConditionalOnResource : 指定的文件存在时起效
@ConditionalOnJndi : 指定的JNDI存在时起效
@ConditionalOnJava : 指定的Java版本存在时起效
@ConditionalOnWebApplication : Web应用环境下起效
@ConditionalOnNotWebApplication : 非Web应用环境下起效
SpringBoot中EnableAutoConfiguration
实现的关键在于引入了AutoConfigurationImportSelector
,其核心逻辑为selectImports
方法,逻辑大致如下:
-
从配置文件META-INF/spring.factories加载所有可能用到的自动配置类;
-
去重,并将exclude和excludeName属性携带的类排除;
-
过滤,将满足条件(
@Conditional
)的自动配置类返回;
@ComponentScan
这个是 Spring 框架的注解,它用来指定组件扫描路径,如果用这个注解,它的值必须包含整个工程中全部需要扫描的路径。因为它会覆盖 SpringBootApplication
的默认扫描路径,导致其失效。
0x02_@Configuration
这个注解严格来说是spring的注解。
源码:
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface Configuration {
@AliasFor(annotation = Component.class)
String value() default "";
boolean proxyBeanMethods() default true;
}
下面详细说明proxyBeanMethods属性:
首先发现@Configuration
注解上打了一个@component
注解
spring会扫描所有@component注解的类及其子类(包括@Configuration注解声明的类),认为这些类是bean, 并且把这些bean对应的beanDefinition放到容器中进行管理。BeanDefinition是对bean的描述,里边存有bean的名称,Class等基本信息
在获取到所有的bean defenition之后,Spring会有一些post process执行,其中一个就是ConfigurationClassPostProcessor
, 在这里,Spring会遍历所有的bean definition, 如果发现其中有标记了@Configuration注解的,会对这个类进行CGLIB代理,生成一个代理的类,并且把这个类设置到BeanDefenition的Class属性中。当需要拿到这个bean的实例的时候,会从这个class属性中拿到的Class对象进行反射,那么最终反射出来的是代理增强后的类
代理中对方法进行了增强?在哪方面进行了增强?对于@Bean标记的方法,返回的都是一个bean,在增强的方法中,Spring会先去容器中查看一下是否有这个bean的实例了,如果有了的话,就返回已有对象,没有的话就创建一个,然后放到容器中
看一个简单的案例:
package com.bones.config;
import com.bones.pojo.User;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration(proxyBeanMethods = false)
public class CommonConfig {
@Bean
public User user1(){
user2();
return new User(1,"root","ok");
}
@Bean
public User user2(){
System.out.println("user2被调用");
return new User(2,"admin","ok");
}
}
proxyBeanMethods
默认为true,表示cglib会为@Configuration生成一个代理类,因此而在user1中调用user2方法时,会通过代理方法从IOC容器中去获取,这样就是单例的。运行的时候,控制台只打印了一次“user2被调用”就证明了这一点
但是如果将proxyBeanMethods设为false,则表示不生成代理,那么user1中调用user2,会再生成一个对象而不是从IOC容器中获取,这样能提高性能,也造成了多例。运行时控制台会打印两次“user2被调用”,如下图所示。
由这一点拓展的知识点:
Full 全模式,Lite 轻量级模式
Full(proxyBeanMethods = true) :proxyBeanMethods参数设置为true时即为:Full 全模式。 该模式下注入容器中的同一个组件无论被取出多少次都是同一个bean实例,即单实例对象,在该模式下SpringBoot每次启动都会判断检查容器中是否存在该组件(找代理,启动慢,单例)
Lite(proxyBeanMethods = false) :proxyBeanMethods参数设置为false时即为:Lite 轻量级模式。该模式下注入容器中的同一个组件无论被取出多少次都是不同的bean实例,即多实例对象,在该模式下SpringBoot每次启动会跳过检查容器中是否存在该组件(不代理,启动快,多例)
什么时候用Full全模式,什么时候用Lite轻量级模式?
当在你的同一个Configuration配置类中,注入到容器中的bean实例之间有依赖关系时,建议使用Full全模式(在上面的例子中,就是存在依赖关系的,user1依赖user2,建议将proxyBeanMethods设置为true,IDEA也是这么提示的)
当在你的同一个Configuration配置类中,注入到容器中的bean实例之间没有依赖关系时,建议使用Lite轻量级模式,以提高springboot的启动速度和性能
proxyBeanMethods 属性默认值是 true, 也就是说该配置类会被代理(CGLIB),在同一个配置文件中调用其它被 @Bean
注解标注的方法获取对象时会直接从 IOC 容器之中获取
proxyBeanMethods设置为 true,那么直接调用方法获取 bean,不会创建新的 bean,而是会走 bean 的生命周期的行为。