@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);
    }
}
相关推荐
七夜zippoe2 分钟前
Java技术未来展望:GraalVM、Quarkus、Helidon等新趋势探讨
java·开发语言·python·quarkus·graaivm·helidon
枫叶落雨2223 分钟前
ClassPathXmlApplicationContext
java·开发语言
草莓熊Lotso3 分钟前
【Linux 线程进阶】进程 vs 线程资源划分 + 线程控制全详解
java·linux·运维·服务器·数据库·c++·mysql
gelald10 分钟前
Spring Boot - 自动配置原理
java·spring boot·后端
hssfscv12 分钟前
软件设计师下午题六——Java的各种设计模式
java·算法·设计模式
希望永不加班23 分钟前
SpringBoot 集成测试:@SpringBootTest 与 MockMvc
java·spring boot·后端·log4j·集成测试
enAn_27 分钟前
对照片和视频文件名,程序追加日期,直观看
java·maven
十五年专注C++开发33 分钟前
Oat++: 一个轻量级、高性能、零依赖的 C++ Web 框架
开发语言·c++·web服务·oatpp
yaaakaaang34 分钟前
六、适配器模式
java·适配器模式
陈天伟教授35 分钟前
心电心音同步分析-案例:原型设计一
开发语言·人工智能·python·语言模型·架构