Java @Bean
随着SpringBoot的流行,现在更多采用基于注解式的配置从而替换掉了基于XML的配置,主要探讨基于注解的@Bean以及和其他注解的使用;

@Bean 基础概念

  • @Bean:Spring的@Bean注解用于告诉方法,产生一个Bean对象,然后这个Bean对象交给Spring管理。产生这个Bean对象的方法Spring只会调用一次,随后这个Spring将会将这个Bean对象放在自己的IOC容器中;
  • SpringIOC 容器管理一个或者多个bean,这些bean都需要在@Configuration注解下进行创建,在一个方法上使用@Bean注解就表明这个方法需要交给Spring进行管理;
  • @Bean是一个方法级别上的注解,主要用在@Configuration注解的类里,也可以用在@Component注解的类里。添加的bean的id为方法名;
  • 使用Bean时,即是把已经在xml文件中配置好的Bean拿来用,完成属性、方法的组装;比如@Autowired@Resource,可以通过byTYPE(@Autowired)byNAME(@Resource)的方式获取Bean;
  • 注册Bean时,@Component@Repository@Controller@Service@Configration这些注解都是把要实例化的对象转化成一个Bean,放在IoC容器中,等要用的时候,它会和上面的@Autowired@Resource配合到一起,把对象、属性、方法完美组装;
  • @Configuration@Bean结合使用:@Configuration可理解为用Spring的时候xml里面的标签,@Bean可理解为用Spring的时候xml里面的标签;

快速搭建一个Maven项目并配置好所需要的Spring 依赖

  1. <dependency>
  2. <groupId>org.springframework</groupId>
  3. <artifactId>spring-context</artifactId>
  4. <version>4.3.13.RELEASE</version>
  5. </dependency>
  6. <dependency>
  7. <groupId>org.springframework</groupId>
  8. <artifactId>spring-beans</artifactId>
  9. <version>4.3.13.RELEASE</version>
  10. </dependency>
  11. <dependency>
  12. <groupId>org.springframework</groupId>
  13. <artifactId>spring-core</artifactId>
  14. <version>4.3.13.RELEASE</version>
  15. </dependency>
  16. <dependency>
  17. <groupId>org.springframework</groupId>
  18. <artifactId>spring-web</artifactId>
  19. <version>4.3.13.RELEASE</version>
  20. </dependency>

在src根目录下创建一个AppConfig的配置类,这个配置类也就是管理一个或多个bean 的配置类,并在其内部声明一个myBean的bean,并创建其对应的实体类

  1. @Configuration
  2. public class AppConfig {
  3. // 使用@Bean 注解表明myBean需要交给Spring进行管理
  4. // 未指定bean 的名称,默认采用的是 "方法名" + "首字母小写"的配置方式
  5. @Bean
  6. public MyBean myBean(){
  7. return new MyBean();
  8. }
  9. }
  10. public class MyBean {
  11. public MyBean(){
  12. System.out.println("MyBean Initializing");
  13. }
  14. }

然后再创建一个测试类SpringBeanApplicationTests,测试上述代码的正确性

  1. public class SpringBeanApplicationTests {
  2. public static void main(String[] args) {
  3. ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
  4. context.getBean("myBean");
  5. }
  6. }

输出 : MyBean Initializing

深入了解@Bean注解的源代码

  1. @Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
  2. @Retention(RetentionPolicy.RUNTIME)
  3. @Documented
  4. public @interface Bean {
  5. @AliasFor("name")
  6. String[] value() default {};
  7. @AliasFor("value")
  8. String[] name() default {};
  9. Autowire autowire() default Autowire.NO;
  10. String initMethod() default "";
  11. String destroyMethod() default AbstractBeanDefinition.INFER_METHOD;
  12. }

@Bean的属性:

  • value:bean别名和name是相互依赖关联的,value,name如果都使用的话值必须要一致;
  • name:bean名称,如果不写会默认为注解的方法名称;
  • autowire:自定装配默认是不开启的,建议尽量不要开启,因为自动装配不能装配基本数据类型、字符串、数组等,这是自动装配设计的局限性,并且自动装配不如依赖注入精确;
  • initMethod:bean的初始化之前的执行方法,该参数一般不怎么用,因为完全可以在代码中实现;
  • destroyMethod:默认使用javaConfig配置的bean,如果存在close或者shutdown方法,则在bean销毁时会自动执行该方法,如果不想执行该方法,则添加@Bean(destroyMethod="")来防止出发销毁方法;

如果发现销毁方法没有执行,原因是bean销毁之前程序已经结束了,可以手动close下如下:

  1. AnnotationConfigApplicationContext applicationContext2 = new AnnotationConfigApplicationContext(MainConfig.class);
  2. User bean2 = applicationContext2.getBean(User.class);
  3. System.out.println(bean2);
  4. //手动执行close方法
  5. applicationContext2.close();

运行结果如下:

  1. 初始化用户bean之前执行
  2. User [userName=张三, age=26]
  3. bean销毁之后执行

可以发现@baen注解的@TargetElementType.METHODElementType.ANNOTATION_TYPE也就说@Bean注解可以在使用在方法上,以及一个注释类型声明

@Bean 注解与其他注解一起使用

@Bean注解不止这几个属性,它还能和其他的注解一起配合使用。
@Profile 注解
@Profile的作用是把一些meta-data进行分类,分成Active和InActive这两种状态,然后可以选择在active 和在Inactive这两种状态下配置bean,在Inactive状态通常的注解有一个!操作符,通常写为:@Profile("!p"),这里的p是Profile的名字。
三种设置方式:

  • 可以通过ConfigurableEnvironment.setActiveProfiles()以编程的方式激活。
  • 可以通过AbstractEnvironment.ACTIVE_PROFILES_PROPERTY_NAME(spring.profiles.active)属性设置为JVM属性。
  • 作为环境变量,或作为web.xml 应用程序的Servlet上下文参数。也可以通过@ActiveProfiles 注解在集成测试中以声明方式激活配置文件。

作用域:

  • 作为类级别的注解在任意类或者直接与@Component 进行关联,包括@Configuration
  • 作为原注解,可以自定义注解
  • 作为方法的注解作用在任何方法

注意:

  • 如果一个配置类使用了Profile 标签或者@Profile 作用在任何类中都必须进行启用才会生效,如果@Profile({“p1”,"!p2"}) 标识两个属性,那么p1 是启用状态 而p2 是非启用状态的。

例如:

  1. @Profile("dev")
  2. public @Bean("activityMongoFactory")
  3. MongoDbFactory activityMongoFactoryDev(MongoClient activityMongo) {
  4. return new SimpleMongoDbFactory(activityMongo, stringValueResolver.resolveStringValue("${mongodb.dev.database}"));
  5. }

@Scope 注解

在Spring中对于bean的默认处理都是单例的,通过上下文容器.getBean方法拿到bean容器,并对其进行实例化,这个实例化的过程其实只进行一次,即多次getBean 获取的对象都是同一个对象,也就相当于这个bean的实例在IOC容器中是public的,对于所有的bean请求来讲都可以共享此bean。
如果不想把这个bean被所有的请求共享或者说每次调用都想让它生成一个新的bean实例,该怎么实现呢?

bean的多个实例

bean的非单例原型范围会使每次发出对该特定bean的请求时都创建新的bean实例,也就是说,bean被注入另一个bean,或者通过对容器的getBean()方法调用来请求它。

下面,通过一个示例来说明bean的多个实例

新建一个ConfigScope配置类,用来定义多例的bean

  1. @Configuration
  2. public class ConfigScope {
  3. /**
  4. * 为myBean起两个名字,b1 和 b2
  5. * @Scope 默认为 singleton,但是可以指定其作用域
  6. * prototype 是多例的,即每一次调用都会生成一个新的实例。
  7. */
  8. @Bean({"b1","b2"})
  9. @Scope("prototype")
  10. public MyBean myBean(){
  11. return new MyBean();
  12. }
  13. }

注意:prototype代表bean对象的定义为任意数量的对象实例,所以指定为prototype属性可以定义为多例,也会每一次调用都会生成一个新的实例。
2021-07-28-22-23-31-333047.png
singletonprototype 一般都用在普通的Java项目中,而requestsessionapplicationwebsocket都用于web应用中。

@Lazy 注解

表明一个bean 是否延迟加载,可以作用在方法上,表示这个方法被延迟加载;可以作用在@Component (或者由@Component 作为原注解) 注释的类上,表明这个类中所有的bean 都被延迟加载。
如果没有@Lazy注释,或者@Lazy 被设置为false,那么该bean 就会急切渴望被加载;除了上面两种作用域,@Lazy 还可以作用在@Autowired@Inject注释的属性上,在这种情况下,它将为该字段创建一个惰性代理,作为使用ObjectFactoryProvider的默认方法。
下面来演示一下:

  1. @Lazy
  2. @Configuration
  3. @ComponentScan(basePackages = "com.spring.configuration.pojo")
  4. public class AppConfigWithLazy {
  5. @Bean
  6. public MyBean myBean(){
  7. System.out.println("myBean Initialized");
  8. return new MyBean();
  9. }
  10. @Bean
  11. public MyBean IfLazyInit(){
  12. System.out.println("initialized");
  13. return new MyBean();
  14. }
  15. }

@Primary 注解

指示当多个候选者有资格自动装配依赖项时,应优先考虑bean。此注解在语义上就等同于在Spring XML中定义的bean 元素的primary属性。
注意:除非使用component-scanning进行组件扫描,否则在类级别上使用@Primary不会有作用。如果@Primary 注解定义在XML中,那么@Primary 的注解元注解就会忽略,相反的话就可以优先使用。
@Primary 的两种使用方式:

  • @Bean 一起使用,定义在方法上,方法级别的注解
  • @Component 一起使用,定义在类上,类级别的注解

新建一个AppConfigWithPrimary类,在方法级别上定义@Primary注解

  1. @Configuration
  2. public class AppConfigWithPrimary {
  3. @Bean
  4. public MyBean myBeanOne(){
  5. return new MyBean();
  6. }
  7. @Bean
  8. @Primary
  9. public MyBean myBeanTwo(){
  10. return new MyBean();
  11. }
  12. }

上面代码定义了两个bean ,其中myBeanTwo@Primary进行标注,表示它首先会进行注册,使用测试类进行测试。