@Scope失效问题

单例的Bean注入其它类型的@Scope属性有可能会失效,针对失效的情况,我们可以采取四种解决办法。

一.失效场景

1.准备多例Bean

java 复制代码
package com.example.springdemo.demos.a27;

import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;

/**
 * @author zhou
 * @version 1.0
 * @description TODO
 * @date 2025/10/15 21:56
 */
@Scope("prototype")
@Component
public class F1 {
}

2.E中注入F1

java 复制代码
package com.example.springdemo.demos.a27;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Component;

/**
 * @author zhou
 * @version 1.0
 * @description TODO
 * @date 2025/10/15 21:56
 */
@Component
public class E {
    @Autowired
    private  F1 f1;
    public F1 getF1(){
        return f1;
    }
}

3.容器中获取对象

java 复制代码
package com.example.springdemo.demos.a27;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.ComponentScan;

/**
 * @author zhou
 * @version 1.0
 * @description TODO
 * @date 2025/10/15 21:54
 */
@ComponentScan("com.example.springdemo.demos.a27")
public class a27Application {
    private static final Logger log = LoggerFactory.getLogger(a27Application.class);

    public static void main(String[] args) {
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(a27Application.class);
        E e = context.getBean(E.class);
        log.info("{}",e.getF1().getClass());
        log.info("{}",e.getF1());
        log.info("{}",e.getF1());
        log.info("{}",e.getF1());

    }
}

结果:

通过依赖注入的F1对象多次获取后都是同一个,Scope作用失效了。分析其原因是对于单例对象E来讲,依赖注入仅仅发生了一次,后续再也没有使用多例的F1,所以多次获取都是同一个对象。

二.解决办法

下面提供四种解决方法。

1.添加@Lazy注解

java 复制代码
package com.example.springdemo.demos.a27;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Component;

/**
 * @author zhou
 * @version 1.0
 * @description TODO
 * @date 2025/10/15 21:56
 */
@Component
public class E {
    @Lazy
    @Autowired
    private  F1 f1;
    public F1 getF1(){
        return f1;
    }
}

结果每次获取出不同的F1对象。使用@Lazy注解之后,注入便是F1的代理对象(注意看输出的F1对象的类型),e.getF1()使用了F1代理类的toString(),而每次使用代理对象的方法都会创建新的F1对象。

2.在Scope属性中设置proxyMode

设置proxyMode的值为ScopedProxyMode.TARGET_CLASS,可以为F1对象生成代理对象,与上面方法类似。

java 复制代码
package com.example.springdemo.demos.a27;

import org.springframework.context.annotation.Scope;
import org.springframework.context.annotation.ScopedProxyMode;
import org.springframework.stereotype.Component;

/**
 * @author zhou
 * @version 1.0
 * @description TODO
 * @date 2025/10/15 21:56
 */
@Scope(value = "prototype",proxyMode = ScopedProxyMode.TARGET_CLASS)
@Component
public class F1 {
}

3.使用ObjectFactory包装F1

java 复制代码
package com.example.springdemo.demos.a27;

import org.springframework.beans.factory.ObjectFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Component;

/**
 * @author zhou
 * @version 1.0
 * @description TODO
 * @date 2025/10/15 21:56
 */
@Component
public class E {
    @Autowired
    private ObjectFactory<F1> f1;
    public F1 getF1(){
        return f1.getObject();
    }
}

通过这种方式每次获取F1,工厂都可以生成多例对象。

4.注入ApplicationContext容器

每次从容器中获取F1对象自然是多例的。

java 复制代码
package com.example.springdemo.demos.a27;

import org.springframework.beans.factory.ObjectFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.Lazy;
import org.springframework.stereotype.Component;

/**
 * @author zhou
 * @version 1.0
 * @description TODO
 * @date 2025/10/15 21:56
 */
@Component
public class E {
    @Autowired
    private ApplicationContext context;
    public F1 getF1(){
        return context.getBean(F1.class);
    }
}
相关推荐
xyq2024几秒前
Matplotlib 绘图线
开发语言
独断万古他化2 分钟前
【Spring 原理】Bean 的作用域与生命周期
java·后端·spring
m0_694845574 分钟前
tinylisp 是什么?超轻量 Lisp 解释器编译与运行教程
服务器·开发语言·云计算·github·lisp
*小海豚*8 分钟前
在linux服务器上DNS正常,但是java应用调用第三方解析域名报错
java·linux·服务器
春日见9 分钟前
如何创建一个PR
运维·开发语言·windows·git·docker·容器
C++ 老炮儿的技术栈12 分钟前
VS2015 + Qt 实现图形化Hello World(详细步骤)
c语言·开发语言·c++·windows·qt
派葛穆19 分钟前
Python-批量安装依赖
开发语言·python
撩得Android一次心动23 分钟前
Android LiveData 全面解析:使用Java构建响应式UI【源码篇】
android·java·android jetpack·livedata
组合缺一27 分钟前
Solon AI (Java) v3.9 正式发布:全能 Skill 爆发,Agent 协作更专业!仍然支持 java8!
java·人工智能·ai·llm·agent·solon·mcp
MSTcheng.31 分钟前
【C++】C++11新特性(二)
java·开发语言·c++·c++11