spring中的bean是单例还是多例?如何保证并发安全?

本文探讨了Spring中Controller默认的单例模式可能导致的并发安全问题。通过示例展示了当在Controller中使用非静态成员变量时,不同请求可能会共享同一变量状态,导致预期结果错误。提出了四种解决方案:避免在Controller中定义成员变量、使用@Scope(prototype)创建多例、利用ThreadLocal实现线程隔离以及使用并发安全类。同时,详细介绍了Spring Bean的几种作用域,包括singleton、prototype、request、session和globalsession。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

spring中的controller默认是单例的,不要使用非静态的成员变量,否则会发生并发安全性问题。

@RestController
public class ScopeController {

    private int num = 0;

    @GetMapping("/testScope")
    public Object testScope() {
        System.out.println(++num);
        return num;
    }

    @GetMapping("/testScope2")
    public Object testScope2() {
        System.out.println(++num);
        return num;
    }

}

1、首先访问testScope,得到返回1;
2、然后访问testScope2,得到返回2;此时不是想要的结果,num被修改了。
因为controller是单例不安全的,所有request请求都访问同一个controller时,成员变量是公用的,如果某个请求修改了这个变量,那么在其他的请求中可以获取修改过的变量。

解决方案
1、不要在controller中定义成员变量;
2、若必须定义一个非静态成员变量时,通过注解@Scope(“prototype”)将其设置为多例模式,则每次请求都创建新的controller;
测试结果:每次请求获得的返回都是1。
3、使用ThreadLocal变量;
测试结果:连续请求得到的返回有1、2、3等,可以看到服务器默认线程池大小为10,这10个可以被request请求复用,所以会出现如上的结果;

@RestController
public class ScopeController {

    private ThreadLocal<Integer> num = new ThreadLocal<>();

    @GetMapping("/testScope")
    public Object testScope() {
        if (num.get() == null) {
            num.set(0);
        }
        num.set(num.get().intValue() + 1);
        return num.get() + Thread.currentThread().getName();
    }

    @GetMapping("/testScope2")
    public Object testScope2() {
        if (num.get() == null) {
            num.set(0);
        }
        num.set(num.get().intValue() + 1);
        return num.get() + Thread.currentThread().getName();
    }

}

ThreadLocal可以达到线程隔离,但还是无法达到并发安全。
4、使用并发安全类;

spring bean的作用域如下;

  • singleton:单例模式,当spring创建applicationContext容器的时候,spring会欲初始化所有的该作用域实例;
  • prototype:原型模式,每次通过getBean获取该bean就会新产生一个实例,创建后spring将不再对其管理;
  • request:每次请求都新产生一个实例,和prototype不同就是创建后,接下来的管理,spring依然在监听;
  • session,每次会话都新产生一个实例,和prototype不同就是创建后,接下来的管理,spring依然在监听;
  • global session:全局的web域,类似于servlet中的application;
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值