欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Spring如何解决单例bean线程不安全的问题

程序员文章站 2022-06-22 14:00:18
首先我们应该知道线程安全问题一般发生在成员变量上,这是为什么啦?因为成员变量是存放在堆内存中,而堆内存又是线程共享的,这就造成了线程安全问题因为spring中的bean默认是单例的,所以在定义成员变量...

首先我们应该知道线程安全问题一般发生在成员变量上,这是为什么啦?

因为成员变量是存放在堆内存中,而堆内存又是线程共享的,这就造成了线程安全问题

因为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模式去启动,打断点。不需要执行完程序,然后再次调用该接口。或者多次调用该接口,便会出现以下控制台所示的结果。

Spring如何解决单例bean线程不安全的问题

下面我们就来讨论下解决这个线程不安全的问题的办法

解决方式一:

在对应的类名上加上该注解@scope("prototype"),表示每次调用该接口都会生成一个新的bean。下图示例

Spring如何解决单例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管理器是“绝对的线程安全”。

以上这篇spring如何解决单例bean线程不安全的问题就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。