Spring注解 @Scope详解

项目中用到@Scope该注解,然后出现一些情况,所以总结下

下面是@Scope注解的源码

@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Scope {

	/**
	 * Alias for {@link #scopeName}.
	 * @see #scopeName
	 */
	@AliasFor("scopeName")
	String value() default "";

	/**
	 * Specifies the name of the scope to use for the annotated component/bean.
	 * <p>Defaults to an empty string ({@code ""}) which implies
	 * {@link ConfigurableBeanFactory#SCOPE_SINGLETON SCOPE_SINGLETON}.
	 * @since 4.2
	 * @see ConfigurableBeanFactory#SCOPE_PROTOTYPE
	 * @see ConfigurableBeanFactory#SCOPE_SINGLETON
	 * @see org.springframework.web.context.WebApplicationContext#SCOPE_REQUEST
	 * @see org.springframework.web.context.WebApplicationContext#SCOPE_SESSION
	 * @see #value
	 */
	@AliasFor("value")
	String scopeName() default "";

	/**
	 * Specifies whether a component should be configured as a scoped proxy
	 * and if so, whether the proxy should be interface-based or subclass-based.
	 * <p>Defaults to {@link ScopedProxyMode#DEFAULT}, which typically indicates
	 * that no scoped proxy should be created unless a different default
	 * has been configured at the component-scan instruction level.
	 * <p>Analogous to {@code <aop:scoped-proxy/>} support in Spring XML.
	 * @see ScopedProxyMode
	 */
	ScopedProxyMode proxyMode() default ScopedProxyMode.DEFAULT;
}

该注解的value属性有5个值:

@Scope(value=ConfigurableBeanFactory.SCOPE_PROTOTYPE)多例,在每次注入的时候回自动创建一个新的bean实例

@Scope(value=ConfigurableBeanFactory.SCOPE_SINGLETON)(默认)单例模式,在整个应用中只能创建一个实例

@Scope(value=WebApplicationContext.SCOPE_GLOBAL_SESSION)全局session

@Scope(value=WebApplicationContext.SCOPE_APPLICATION)在一个web应用中只创建一个实例

@Scope(value=WebApplicationContext.SCOPE_REQUEST)在一个请求中创建一个实例

@Scope(value=WebApplicationContext.SCOPE_SESSION)每次创建一个会话中创建一个实例

proxyMode属性3个值:

proxyMode=ScopedProxyMode.INTERFACES创建一个JDK代理模式

proxyMode=ScopedProxyMode.TARGET_CLASS基于类的代理模式

proxyMode=ScopedProxyMode.NO(默认)不进行代理

结构目录
在这里插入图片描述

在单例模式下:

//control:
@RestController
@RequestMapping("/test")
public class TestController {

    @Autowired
    private AService aService;

    @RequestMapping("/say")
    public void say(){
        aService.sayHello();
    }
}

//service:
public interface AService {
    void sayHello();
}
@Service
//@Scope(value= ConfigurableBeanFactory.SCOPE_REQUEST,proxyMode = ScopedProxyMode.INTERFACES) //之后用到
public class AServiceImpl implements AService {
    @Autowired
    private TestC testC;

    @Override
    public void sayHello() {
        System.out.println(Thread.currentThread()+"service对象"+this);
        testC.say();
    }
}

//实体
@Component
@Getter
@Setter
public class TestC {
    private static int x = 0;
    private int i = 0;
    public void say(){
        System.out.println("i:"+i+++" | x:"+x++);
    }
}

//测试类:
@SpringBootTest(webEnvironment =SpringBootTest.WebEnvironment.MOCK,classes = DemoApplication.class)
@AutoConfigureMockMvc
public class DemoApplicationTests {

    private ExecutorService service = Executors.newFixedThreadPool(5);

    @Autowired
    private MockMvc mockMvc;

    @Test
    public void test() throws InterruptedException {

        CountDownLatch countDownLatch = new CountDownLatch(5);
        for (int i = 0; i < 5; i++) {
            service.execute(new RunableTask(countDownLatch,mockMvc));
        }
        countDownLatch.await();
    }
}

class RunableTask implements Runnable {

    private CountDownLatch countDownLatch;
    private MockMvc mockMvc;
    public RunableTask(CountDownLatch countDownLatch,MockMvc mockMvc) {
        this.countDownLatch = countDownLatch;
        this.mockMvc = mockMvc;
    }

    @Override
    public void run() {
        try {
            MvcResult mvcResult = mockMvc.perform(get("/test/say").contentType(MediaType.APPLICATION_JSON)).andReturn();
            mvcResult.getResponse().getContentAsString();
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            countDownLatch.countDown();
        }
    }

运行test方法:
在这里插入图片描述
这里使用多线程是为了模拟并发访问controller

假设: 现在的业务需要我们每次请求的是后testC都是新的,这里我们就需要用到多例了,在TestC类上注解@Scope(value=ConfigurableBeanFactory.SCOPE_PROTOTYPE)
运行test方法发现结果跟单例的时候一样!!i的值还是累加在一起
在这里插入图片描述
这是因为你的controller、service都是单例,testC注入到service后,尽管你将testC注解为多例,但是service中的testC还是原来的对象
解决方法是将TestC类上的注解改为@Scope(value= ConfigurableBeanFactory.SCOPE_PROTOTYPE,proxyMode = ScopedProxyMode.TARGET_CLASS)
运行test方法 结果是我们想要的!!
在这里插入图片描述
我们试下作用域为request情况下:
在AServiceImpl类上注解@Scope(value= WebApplicationContext.SCOPE_REQUEST)运行发现报错

Caused by: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name ‘testController’: Unsatisfied dependency expressed through field ‘aService’; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name ‘AServiceImpl’: Scope ‘request’ is not active for the current thread; consider defining a scoped proxy for this bean if you intend to refer to it from a singleton; nested exception is java.lang.IllegalStateException: No thread-bound request found: Are you referring to request attributes outside of an actual web request, or processing a request outside of the originally receiving thread? If you are actually operating within a web request and still receive this message, your code is probably running outside of DispatcherServlet: In this case, use RequestContextListener or RequestContextFilter to expose the current request.

我理解的意思是:因为Controller需要注入一个service,但是该是request作用域,因此可以为每个请求创建一个新实例。但是,在启动时,没有这样的请求,因此AService无法注入Bean,因为它不存在。所以报错了!!其中singleton和prototype用于一半情况,而后面3个用于web项目。

所以我们将AServiceImpl类上注解改为@Scope(value= ConfigurableBeanFactory.SCOPE_REQUEST,proxyMode = ScopedProxyMode.INTERFACES)

运行代码 结果是我们要的!!
在这里插入图片描述
说明每次请求的时候都创建一个AServiceImpl的bean!

之后的session、global_session大同小异,大家自己取试下吧,我就不介绍拉!

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
东南亚位于我国倡导推进的“一带一路”海陆交汇地带,作为当今全球发展最为迅速的地区之一,近年来区域内生产总实现了显著且稳定的增长。根据东盟主要经济体公布的最新数据,印度尼西亚2023年国内生产总(GDP)增长5.05%;越南2023年经济增长5.05%;马来西亚2023年经济增速为3.7%;泰国2023年经济增长1.9%;新加坡2023年经济增长1.1%;柬埔寨2023年经济增速预计为5.6%。 东盟国家在“一带一路”沿线国家中的总体GDP经济规模、贸易总额与国外直接投资均为最大,因此有着举足轻重的地位和作用。当前,东盟与中国已互相成为双方最大的交易伙伴。中国-东盟贸易总额已从2013年的443亿元增长至 2023年合计超逾6.4万亿元,占中国外贸总的15.4%。在过去20余年中,东盟国家不断在全球多变的格局里面临挑战并寻求机遇。2023东盟国家主要经济体受到国内消费、国外投资、货币政策、旅游业复苏、和大宗商品出口价企稳等方面的提振,经济显现出稳步增长态势和强韧性的潜能。 本调研报告旨在深度挖掘东南亚市场的增长潜力与发展机会,分析东南亚市场竞争态势、销售模式、客户偏好、整体市场营商环境,为国内企业出海开展业务提供客观参考意见。 本文核心内容: 市场空间:全球行业市场空间、东南亚市场发展空间。 竞争态势:全球份额,东南亚市场企业份额。 销售模式:东南亚市场销售模式、本地代理商 客户情况:东南亚本地客户及偏好分析 营商环境:东南亚营商环境分析 本文纳入的企业包括国外及印尼本土企业,以及相关上下游企业等,部分名单 QYResearch是全球知名的大型咨询公司,行业涵盖各高科技行业产业链细分市场,横跨如半导体产业链(半导体设备及零部件、半导体材料、集成电路、制造、封测、分立器件、传感器、光电器件)、光伏产业链(设备、硅料/硅片、电池片、组件、辅料支架、逆变器、电站终端)、新能源汽车产业链(动力电池及材料、电驱电控、汽车半导体/电子、整车、充电桩)、通信产业链(通信系统设备、终端设备、电子元器件、射频前端、光模块、4G/5G/6G、宽带、IoT、数字经济、AI)、先进材料产业链(金属材料、高分子材料、陶瓷材料、纳米材料等)、机械制造产业链(数控机床、工程机械、电气机械、3C自动化、工业机器人、激光、工控、无人机)、食品药品、医疗器械、农业等。邮箱:market@qyresearch.com
Spring是一个Java开发框架,提供了许多注解来简化开发。下面是Spring中常用的注解详解: 1. @Autowired:自动装配,用于自动注入依赖项。 2. @Component:用于标记一个类为Spring容器管理的组件。 3. @Controller:用于标记一个类为Spring MVC控制器。 4. @Service:用于标记一个类为服务层组件。 5. @Repository:用于标记一个类为数据访问层组件。 6. @RequestMapping:用于映射请求到处理程序方法。 7. @PathVariable:用于获取请求URL中的变量。 8. @RequestParam:用于获取请求参数的。 9. @ResponseBody:用于将返回直接写入响应体中。 10. @RestController:用于标记一个类为Spring MVC控制器,并且默认所有方法都返回JSON格式的数据。 11. @Configuration:用于标记一个类为配置类,相当于一个XML配置文件。 12. @Bean:用于在配置类中声明一个Bean。 13. @Value:用于获取配置文件中的属性。 14. @Qualifier:用于指定自动装配时使用的Bean名称。 15. @Scope:用于指定Bean的作用域,例如singleton、prototype等。 16. @Transactional:用于标记一个方法为事务方法。 17. @Async:用于标记一个方法为异步方法。 18. @PostConstruct:用于标记一个方法为Bean初始化方法。 19. @PreDestroy:用于标记一个方法为Bean销毁方法。 以上是Spring中常见的注解详解,使用这些注解可以使开发更加简单和高效。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值