绝对让你大开眼界的Spring依赖注入问题

开发 前端
本篇文章旨在深入探讨将原型(Prototype)作用域的bean注入到单例(Singleton)作用域bean中的多种策略,并详细剖析每种方法的优缺点。

环境:SpringBoot3.2.5

1. 简介

本篇文章旨在深入探讨将原型(Prototype)作用域的bean注入到单例(Singleton)作用域bean中的多种策略,并详细剖析每种方法的优缺点。在Spring框架中,默认情况下bean是单例的,这意味着它们在应用的生命周期内仅被创建一次并共享。然而,当需要将原型bean(每次请求都创建一个新实例)注入到单例bean中时,就会出现作用域注入问题。如下示例:

@Configuration
public class AppConfig {
  @Bean
  @Scope(BeanDefinition.SCOPE_PROTOTYPE)
  public PrototypeBean prototypeBean() {
    return new PrototypeBean() ;
  }
  @Bean
  public SingletonBean singletonBean() {
    return new SingletonBean() ;
  }
}

在该配置类中,将PrototypeBean定义为原型bean,SingletonBean定义为单例,该单例bean的定义如下:

@Component
public class SingletonBean {


  @Resource
  private PrototypeBean prototypeBean ;  
  public void test() {
    System.out.println(this.prototypeBean) ;
  }
}

在这通过@Resource注解,注入原型bean。接下来我们通过如下的代码进行测试:

ConfigurableApplicationContext context = SpringApplication.run(App.class, args) ;
SingletonBean bean = context.getBean(SingletonBean.class) ;
bean.test() ;
bean.test() ;
bean.test() ;

以上调用三次test方法,我们期望的是每次调用打印的prototypeBean属性都不是同一个,结果如下:

com.pack.PrototypeBean@69fb6037
com.pack.PrototypeBean@69fb6037
com.pack.PrototypeBean@69fb6037

三次打印都是同一个。这种不同作用域的注入该如何解决呢?接下来我将介绍8种方式解决。

2. 解决办法

2.1 单例bean实现ApplicationContextAware

public class SingletonBean implements ApplicationContextAware {
  private ApplicationContext context;
  public void test() {
    System.out.println(this.context.getBean(PrototypeBean.class)) ;
  }
  public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
    this.context = applicationContext ;
  }
}

每调用test方法时都会通过ApplicationContext对象从容器中获取新实例。

缺点:

  • 这种方法有个严重的缺点。它违背了控制权倒置的原则,因为我们直接向容器请求依赖关系。
  • 在SingletonBean类中通过applicationContext获取原型 Bean。这意味着将代码与 Spring 框架耦合。

除了这2缺点,用起来感觉还行啊

2.2 通过jakarta的Provider注入

public class SingletonBean {
  @Resource
  private jakarta.inject.Provider<PrototypeBean> provider ;
  public void test() {
    System.out.println(provider.get()) ;
  }
}

这里通过jakarta.inject包中的Provider接口注入原型bean。每次调用get方法都会从容器中再次获取一个新的实例对象。

2.3 借助Spring提供的ObjectProvider接口

public class SingletonBean {
  @Resource
  private ObjectProvider<PrototypeBean> objectProvider ;
  public void test() {
    System.out.println(objectProvider.getIfAvailable()) ;
  }
}

直接注入ObjectProvider接口对象,该接口提供了丰富的方法,不仅仅可以获取单个对象,也可以获取多个对象(集合),包括如下可用方法:

图片图片

2.4 借助Spring的ObjectFactory接口

public class SingletonBean {
  @Resource
  private ObjectFactory<PrototypeBean> objectFactory ;
  public void test() {
    System.out.println(objectFactory.getObject()) ;
  }
}

该接口与上面ObjectProvider接口挺像,但是ObjectFactory是函数式接口只有一个getObject方法,并且只能返回一个对象。

2.5 使用@Lookup注解

public class SingletonBean {
  
  public void test() {
    System.out.println(get()) ;
  }
  @Lookup
  protected PrototypeBean get() {
    return null ;
  }
}

Spring 会将SingletonBean创建为代理对象并覆盖使用 @Lookup 注解的 get() 方法。每当我们调用get() 方法时,它会返回一个新的 PrototypeBean 实例。

2.6 注册Function类型的Bean

@Bean
public <T> Function<Class<T>, T> targetBean(ApplicationContext context) {
  return t -> context.getBean(t) ;
}

这里注册了Function类型的bean对象,而具体会是什么类型完全由使用的地方决定。

public class SingletonBean {
  @Resource
  private Function<Class<?>, PrototypeBean> targetBean ;
  public void test() {
    System.out.println(targetBean.apply(PrototypeBean.class)) ;
  }
}

通过定义Function类型的bean,这里我们就可以不限制具体是那个类型的原型bean,这就非常的通用了。

2.7 作用域代理

@Component
@Scope(value = BeanDefinition.SCOPE_PROTOTYPE, proxyMode = ScopedProxyMode.TARGET_CLASS)
public class PrototypeBean {
}

在@Scope注解上配置proxyMode=TARGET_CLASS这样配置以后,容器将会为PrototypeBean创建代理对象。也是能够保证每次获取的都是新的实例对象。

2.8 使用@Lazy注解

public class SingletonBean {
  @Resource
  @Lazy
  private PrototypeBean prototypeBean ;
}

通过@Lazy注解后,这里注入的PrototypeBean将会是一个代理对象,这样也可以保证每次使用时都是新的实例对象。

责任编辑:武晓燕 来源: Spring全家桶实战案例源码
相关推荐

2024-06-06 08:06:26

2021-04-03 13:12:43

微信技巧Bug

2018-02-25 08:48:50

百度职业代码

2021-01-10 08:16:25

微信移动应用实用技巧

2014-04-01 11:39:38

集装箱数据中心谷歌

2010-12-20 13:53:10

GoogleWebGL

2020-09-17 20:25:00

人工智能

2009-05-19 10:18:00

机房网络管理

2020-04-06 09:21:04

AI医疗汽车行业

2011-04-13 14:52:13

Qcon

2021-01-12 05:57:49

AI人工智能机器学习

2021-12-23 20:00:59

Firefox浏览器开源

2018-04-25 10:46:05

Linux命令行日历

2021-12-20 10:32:05

IT技术领导者CIO

2024-01-15 07:00:00

2024-12-16 08:11:45

Python系统调用

2021-12-28 11:06:22

Python 开发编程语言

2013-01-22 16:39:44

NFC移动支付

2018-11-23 14:30:19

人工智能人脸识别刷手掌

2015-05-12 10:34:45

点赞
收藏

51CTO技术栈公众号