java - 将原型(prototype)注入(inject)单例(java配置+注释)

标签 java spring scope spring-boot spring-java-config

面试时有人问我有关将原型(prototype)注入(inject)单例的问题。我很难回答,现在我正在努力研究这个问题。

我写了以下代码(pring boot)

bean 1:

@Service
@Scope(value = "prototype")
public class MyValidator {
}

bean 2:

@Service
public class ValidatorHolder {

    @Autowired
    MyValidator myValidator;

    public MyValidator getMyValidator() {
        return myValidator;
    }
}

配置:

@SpringBootApplication
@Configuration
@ComponentScan("com.example.domain")
public class DemoApplication {


    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class, args);
        ApplicationContext context = new AnnotationConfigApplicationContext(DemoApplication.class);
        ValidatorHolder validatorHolder1 = (ValidatorHolder) context.getBean("validatorHolder");
        ValidatorHolder validatorHolder2 = (ValidatorHolder) context.getBean("validatorHolder");
        System.out.println("=====================================");
        System.out.println(validatorHolder1.getMyValidator() == validatorHolder2.getMyValidator());
        System.out.println("=====================================");

    }
}

此代码返回 true。

正如我在阅读 article link 时所理解的那样 可以配置为返回 false。

我可以在我的代码中做什么? (没有 XML)

附注

我尝试重写文章中的代码:

   <bean id="validatorHolder" class="com.example.domain.ValidatorHolder">
        <property name="myValidator" ref="validator"/>
    </bean>

    <bean id="validator" scope="prototype" class="com.example.domain.MyValidator">
        <!-- This instructs the container to proxy the current bean-->
        <aop:scoped-proxy/>
    </bean>

在 main 方法中我编写了以下代码:

ApplicationContext xmlContext = new FileSystemXmlApplicationContext("classpath:applicationContext.xml");
ValidatorHolder validatorHolder21 = (ValidatorHolder) xmlContext.getBean("validatorHolder");
ValidatorHolder validatorHolder22 = (ValidatorHolder) xmlContext.getBean("validatorHolder");
System.out.println("=====================================");
System.out.println(validatorHolder21.getMyValidator() == validatorHolder22.getMyValidator());
System.out.println("=====================================");

无论如何我看到true

附注2

让我们研究 Sean Patrick Floyd 的答案(范围代理,b))

我使用以下主要方法类:

@SpringBootApplication
@ComponentScan("com.example.domain")
public class DemoApplication {


    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class, args);

        ApplicationContext context = new AnnotationConfigApplicationContext(DemoApplication.class);
        ValidatorHolder validatorHolder1 = (ValidatorHolder) context.getBean("validatorHolder");
        ValidatorHolder validatorHolder2 = (ValidatorHolder) context.getBean("validatorHolder");
        System.out.println("=====================================");
        System.out.println(validatorHolder1.getMyValidator() == validatorHolder2.getMyValidator());
        System.out.println("=====================================");         
}

当我运行应用程序时 - 我明白了

org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'validatorHolder' defined in file [D:\freelance\demo\target\classes\com\example\domain\ValidatorHolder.class]: Instantiation of bean failed; nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [com.example.domain.ValidatorHolder]: No default constructor found; nested exception is java.lang.NoSuchMethodException: com.example.domain.ValidatorHolder.<init>()
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.instantiateBean(AbstractAutowireCapableBeanFactory.java:1099)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBeanInstance(AbstractAutowireCapableBeanFactory.java:1044)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:504)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:476)
    at org.springframework.beans.factory.support.AbstractBeanFactory$1.getObject(AbstractBeanFactory.java:303)
    at org.springframework.beans.factory.support.DefaultSingletonBeanRegistry.getSingleton(DefaultSingletonBeanRegistry.java:230)
    at org.springframework.beans.factory.support.AbstractBeanFactory.doGetBean(AbstractBeanFactory.java:299)
    at org.springframework.beans.factory.support.AbstractBeanFactory.getBean(AbstractBeanFactory.java:194)
    at org.springframework.beans.factory.support.DefaultListableBeanFactory.preInstantiateSingletons(DefaultListableBeanFactory.java:755)
    at org.springframework.context.support.AbstractApplicationContext.finishBeanFactoryInitialization(AbstractApplicationContext.java:759)
    at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:480)
    at org.springframework.boot.SpringApplication.refresh(SpringApplication.java:689)
    at org.springframework.boot.SpringApplication.run(SpringApplication.java:321)
    at org.springframework.boot.SpringApplication.run(SpringApplication.java:969)
    at org.springframework.boot.SpringApplication.run(SpringApplication.java:958)
    at com.example.DemoApplication.main(DemoApplication.java:20)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:483)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.java:134)
Caused by: org.springframework.beans.BeanInstantiationException: Failed to instantiate [com.example.domain.ValidatorHolder]: No default constructor found; nested exception is java.lang.NoSuchMethodException: com.example.domain.ValidatorHolder.<init>()
    at org.springframework.beans.factory.support.SimpleInstantiationStrategy.instantiate(SimpleInstantiationStrategy.java:85)
    at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.instantiateBean(AbstractAutowireCapableBeanFactory.java:1092)
    ... 20 common frames omitted
Caused by: java.lang.NoSuchMethodException: com.example.domain.ValidatorHolder.<init>()
    at java.lang.Class.getConstructor0(Class.java:3074)
    at java.lang.Class.getDeclaredConstructor(Class.java:2170)
    at org.springframework.beans.factory.support.SimpleInstantiationStrategy.instantiate(SimpleInstantiationStrategy.java:80)
    ... 21 common frames omitted

PS3

P.S.2 问题与构造函数中缺少 @Autowired 有关

修复此问题后

System.out.println(validatorHolder1.getMyValidator() == validatorHolder2.getMyValidator());

返回真

但如果稍微替换一下 MyValidator 代码:

@Service
@Scope(value="prototype", proxyMode=ScopedProxyMode.TARGET_CLASS)
public class MyValidator {
    Object object = new Object();

    public Object getObject() {
        return object;
    }

    public void setObject(Object object) {
        this.object = object;
    }
}
<小时/>
 System.out.println(validatorHolder1.getMyValidator() == validatorHolder2.getMyValidator());

正确

 System.out.println(validatorHolder1.getMyValidator().getObject() == validatorHolder2.getMyValidator().getObject());

甚至

System.out.println(validatorHolder1.getMyValidator().getObject() == validatorHolder1.getMyValidator().getObject());

最佳答案

您误解了方法注入(inject)技术。您需要使您的 bean 抽象化才能工作:

public class MyValidator {}

public abstract class ValidatorHolder {
    public abstract MyValidator getMyValidator();
}

现在您可以在 XML 中定义 bean,如下所示:

<bean class="com.somepackage.MyValidator" scope="prototype" />
<bean class="com.somepackage.ValidatorHolder">
    <lookup-method name="getMyValidator" bean="myValidator" />
</bean>

在这种情况下,Spring 将创建 ValidatorHolder 的匿名子类,该子类返回其调用的每种类型的原型(prototype) bean(一个新副本)。

使用带注释的服务类,查找方法注入(inject)是不可能的,但这是使用 @Configuration 类实现的方法:

@Configuration
public class MyConfiguration{

    @Bean
    @Scope("prototype")
    public MyValidator myValidator(){
        return new MyValidator();
    }

    @Bean
    public ValidatorHolder validatorHolder(){
        return new ValidatorHolder(){
            @Override public MyValidator getMyValidator(){
                return myValidator();
            }
        };
    }
}

在本例中,您自己创建了 ValidatorHolder 的子类,您可以清楚地看到发生了什么。

但是,只有当您将 bean 和提供者方法抽象化时,这两个版本才有效。

最后一点,可以通过三种不同的方式来定义 spring bean:

  • XML
  • 带有组件的带注释的类(例如 @Service@Component) 扫描
  • @Configuration 类,具有 @Bean 方法。

在您的示例代码中,您混合了这三种样式,这几乎从来都不是一个好主意。选择一种技术并坚持下去。

<小时/>

关于作用域代理,这可以通过所有三种 bean 注册技术来实现。

a) XML

public class MyValidator {}

public class ValidatorHolder {
    private MyValidator myValidator;
    public void setMyValidator(MyValidator myValidator){
        this.myValidator = myValidator;}
    public MyValidator getMyValidator();
}

<bean class="com.somepackage.MyValidator" scope="prototype" />
<bean class="com.somepackage.ValidatorHolder">
    <aop:scoped-proxy />
</bean>

b) 带注释的服务类

@Service @Scope(value="prototype", proxyMode=ScopedProxyMode.TARGET_CLASS)
public class MyValidator {}

@Service
public class ValidatorHolder {
    @Autowired
    public ValidatorHolder(MyValidator myValidator){
        this.myValidator=myValidator;
    }
    private final MyValidator myValidator;
    public MyValidator getMyValidator(){ return myValidator; };
}

c) @Configuration 类,像 XML 版本中的 Bean 类

@Configuration
public class MyConfiguration{

    @Bean
    @Scope("prototype")
    public MyValidator myValidator(){
        return new MyValidator();
    }

    @Bean
    public ValidatorHolder validatorHolder(){
        return new ValidatorHolder(myValidator());
    }

}

请注意,所有代理解决方案将始终返回相同的对象,即代理。但底层功能将委托(delegate)给不同的对象。通过将此代码添加到 MyValidator 来尝试一下:

private int counter = 1;
public int counter(){
    return counter ++;
}

现在,无论您调用此代码的频率如何:

validatorHolder.getMyValidator().counter();

它将始终返回1

关于java - 将原型(prototype)注入(inject)单例(java配置+注释),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33328685/

相关文章:

swift - 如何在 Swift 中从另一个结构编辑一个结构中的数组?

java - 如何在spring中从rest url获取包含问号(?)的参数

java - 我无法将 jpanel 添加到 jframe

spring - MockMvc 和 WebTestClient 有什么区别?

python - python exec() 中的全局变量和局部变量

matlab - matlab中类库的全局变量

java - 比较排序操作中使用静态数组和 java.util.ArrayList 的内存使用情况

java - Hibernate 验证 - 使用 @GroupSequence 对嵌套类进行分组

java - 由于缺少 bean 端点映射,迁移到 Spring WS 2.0 失败?

java - 带有纯 java 配置的 Spring 3.2 @value 注释不起作用,但 Environment.getProperty 有效