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("/getMethod")
    public void getMethod() {
        System.out.println(++num);
    }

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

}

首先访问 http://localhost:8080/getMethod,得到的答案是1;

然再访问 http://localhost:8080/getMethod2,得到的答案是 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;

@Controller
@Scope("prototype")
public class ScopeTestController {

    private int num = 0;

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

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

}

依旧首先访问 http://localhost:8080/getMethod,得到的答案是 1;

然后再访问  http://localhost:8080/getMethod2,得到的答案还是 1。

由此可得出结论

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

总结

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

2、万一必须要定义一个非静态成员变量时候,则通过注解@Scope(“prototype”),将其设置为原型模式(每次通过getBean获取该bean就会新产生一个实例,创建后spring将不再对其管理)

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。
I have a dream : Sandy beach B-J-N.
原文地址:https://www.cnblogs.com/mjtabu/p/14413233.html