Spring如何解决单例bean线程不安全的问题
作者:有趣的灵魂_不世俗的心 时间:2023-12-18 23:50:20
首先我们应该知道线程安全问题一般发生在成员变量上,这是为什么啦?
因为成员变量是存放在堆内存中,而堆内存又是线程共享的,这就造成了线程安全问题
因为Spring中的Bean默认是单例的,所以在定义成员变量时也有可能会发生线程安全问题。下面我们就来研究下如何解决Spring中单例Bean的线程安全问题
@RestController
//@Scope("prototype")
public class BeanController {
private int content=0; //基本类型 线程不安全
private String test=null;//引用类型 线程不安全
@RequestMapping("testBean")
public Object getSercurity(){
System.out.println(content);
System.out.println(test);
content=20;
test="单例模式是不安全的";
return test;
}
问题来了,我们该如何测试线程不安全问题啦?我们需要在程序中用debug模式去启动,打断点。不需要执行完程序,然后再次调用该接口。或者多次调用该接口,便会出现以下控制台所示的结果。
下面我们就来讨论下解决这个线程不安全的问题的办法
解决方式一:
在对应的类名上加上该注解@Scope("prototype"),表示每次调用该接口都会生成一个新的Bean。下图示例
解决方案二 ThreadLocal解决问题
@RestController
//@Scope("prototype")
public class BeanController {
private static ThreadLocal<Integer> content = new ThreadLocal<Integer>() {
@Override
protected Integer initialValue() {
return (int)(Math.random()*10+100);
}
};
private static ThreadLocal<String> test = new ThreadLocal<String>() {
@Override
protected String initialValue() {
return "单例模式是不安全的"+(int)(Math.random()*10+100);
}
};
@RequestMapping("testBean")
public Object getSercurity(){
System.out.println(content.get());
System.out.println(test.get()); System.out.println();
return test.get();
}
}
第三种解决方案:
尽量不要使用成员变量
第四种解决方案:
前提:
该程序是web应用,可以使用Spring Bean的作用域中的request,就是说在类前面加上@Scope("request"),表明每次请求都会生成一个新的Bean对象。
作用于@Scope("prototype")类似。
补充知识:SpringMVC是单例的,高并 * 况下,如何保证性能的?
首先在大家的思考中,肯定有影响的,你想想,单例顾名思义:一个个排队过... 高访问量的时候,你能想象服务器的压力了... 而且用户体验也不怎么好,等待太久~
实质上这种理解是错误的,Java里有个API叫做ThreadLocal,spring单例模式下用它来切换不同线程之间的参数。用ThreadLocal是为了保证线程安全,实际上ThreadLoacal的key就是当前线程的Thread实例。单例模式下,spring把每个线程可能存在线程安全问题的参数值放进了ThreadLocal。这样虽然是一个实例在操作,但是不同线程下的数据互相之间都是隔离的,因为运行时创建和销毁的bean大大减少了,所以大多数场景下这种方式对内存资源的消耗较少,而且并发越高优势越明显。
总的来说就是,单利模式因为大大节省了实例的创建和销毁,有利于提高性能,而ThreadLocal用来保证线程安全性。
另外补充说一句,单例模式是spring推荐的配置,它在高并发下能极大的节省资源,提高服务抗压能力。spring IOC的bean管理器是“绝对的线程安全”。
来源:https://blog.csdn.net/weixin_42324471/article/details/90603651