Controller是单例模式的吗?如何保证线程安全?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Controller是单例模式的吗?如何保证线程安全?相关的知识,希望对你有一定的参考价值。

参考技术A Controller是单例模式的吗?如何保证线程安全?

答:Controller是单例的,也就是说并发请求调用Controller生成的是同一个对象。从线程安全的角度来说,这些线程共享Controller的实例对象。

接下来我们说一下线程安全的问题。首先明确一点,spring的controller是非线程安全的。既然是非线程安全的,那么我们就要注意共享内存中的对象了,这些对象必须要注意线程安全问题。

如何解决线程安全问题呢?

第一种方式,避免定义全局变量。局部变量不存在线程安全问题。

第二种方式,使用ThreadLocal来进行线程隔离。如何使用这里不再赘述。网上有很多例子。

spring的controller是单例还是多例,怎么保证并发的安全

 

controller默认是单例的,不要使用非静态的成员变量,否则会发生数据逻辑混乱。
正因为单例所以不是线程安全的。

我们下面来简单的验证下:

 

package com.riemann.springbootdemo.controller;

import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;


@Controller
public class ScopeTestController {

    private int num = 0;

    @RequestMapping("/testScope")
    public void testScope() {
        System.out.println(++num);
    }

    @RequestMapping("/testScope2")
    public void testScope2() {
        System.out.println(++num);
    }

}

我们首先访问 http://localhost:8080/testScope,得到的答案是1
然后我们再访问 http://localhost:8080/testScope2,得到的答案是 2

得到的不同的值,这是线程不安全的。

 

接下来我们再来给controller增加作用多例 @Scope("prototype")

package com.riemann.springbootdemo.controller;

import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

/**
 * @author riemann
 * @date 2019/07/29 22:56
 */
@Controller
@Scope("prototype")
public class ScopeTestController {

    private int num = 0;

    @RequestMapping("/testScope")
    public void testScope() {
        System.out.println(++num);
    }

    @RequestMapping("/testScope2")
    public void testScope2() {
        System.out.println(++num);
    }

}

 

我们依旧首先访问 http://localhost:8080/testScope,得到的答案是1
然后我们再访问 http://localhost:8080/testScope2,得到的答案还是 1

 

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

解决方案
1、不要在controller中定义成员变量。
2、万一必须要定义一个非静态成员变量时候,则通过注解@Scope(“prototype”),将其设置为多例模式。
3、在Controller中使用ThreadLocal变量

补充说明
spring bean作用域有以下5个:

singleton:单例模式,当spring创建applicationContext容器的时候,spring会欲初始化所有的该作用域实例,加上lazy-init就可以避免预处理;

prototype:原型模式,每次通过getBean获取该bean就会新产生一个实例,创建后spring将不再对其管理;

(下面是在web项目下才用到的)

request:搞web的大家都应该明白request的域了吧,就是每次请求都新产生一个实例,和prototype不同就是创建后,接下来的管理,spring依然在监听;

session:每次会话,同上;

global session:全局的web域,类似于servlet中的application。

 

以上是关于Controller是单例模式的吗?如何保证线程安全?的主要内容,如果未能解决你的问题,请参考以下文章

清晨尝鲜,什么是单例模式?

多线程下的单例模式

springboot默认创建的bean是单实例

高并发下线程安全的单例模式

如何保证 Controller 的并发安全

阿里一面:如何保证 Controller 的并发安全?