Spring - Spring单例Bean是如何保证性能的

Spring单例Bean是如何保证性能的

Spring单例模式与线程安全

Spring框架里的bean或者component,在获取实例时都是默认的单例模式。单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。当多用户同时请求一个服务时,容器会给每一个请求分配一个线程,并使用ThreadLocal,从而保证系统的性能。

ThreadLocal和线程同步机制相比有什么优势呢?

在同步机制中,通过对象的锁机制保证同一时间只有一个线程访问变量。这时该变量是多个线程共享的,使用同步机制要求程序慎密地分析什么时候对变量进行读写,什么时候需要锁定某个对象,什么时候释放对象锁等繁杂的问题,程序设计和编写难度相对较大。 

ThreadLocal则从另一个角度来解决多线程的并发访问。ThreadLocal会为每一个线程提供一个独立的变量副本,从而隔离了多个线程对数据的访问冲突。因为每一个线程都拥有自己的变量副本,从而也就没有必要对该变量进行同步了。ThreadLocal提供了线程安全的共享对象,在编写多线程代码时,可以把不安全的变量封装进ThreadLocal。 

概括起来说,对于多线程资源共享的问题,同步机制采用了“以时间换空间”的方式,而ThreadLocal采用了“以空间换时间”的方式。前者仅提供一份变量,让不同的线程排队访问,而后者为每一个线程都提供了一份变量,因此可以同时访问而互不影响。 

Spring MVC Controller单例陷阱

单例模式下:

@RestController
@RequestMapping("controller")
public class TestController {
    private static int index_s = 0;//静态的
    private int index = 0;//非静态

    @GetMapping("/test")
    public String test() {
        return index_s++ + " | " + index++;
    }
}

结果:

0 | 0
1 | 1
2 | 2
3 | 3
4 | 4

多例模式下:

@RestController
@RequestMapping("controller")
@Scope("prototype")
public class TestController {
    private static int index_s = 0;//静态的
    private int index = 0;//非静态

    @GetMapping("/test")
    public String test() {
        return index_s++ + " | " + index++;
    }
}

结果:

0 | 0
1 | 0
2 | 0
3 | 0
4 | 0

由此可见:

单例是线程不安全的,会导致属性的重复性利用。

1、不要在controller中定义成员变量。

2、万一必须要定义一个非静态成员变量时候,则通过注解@Scope("prototype"),将其设置为多例模式

 

 

参考资料:

https://blog.51cto.com/lavasoft/1394669

https://blog.csdn.net/paladinzh/article/details/88051356 

posted @ 2019-06-08 15:57  Helios_Fz  阅读(2852)  评论(0编辑  收藏  举报