SpringMVC的控制器是单例的吗?

过去的,未来的
2020-03-03 / 0 评论 / 0 点赞 / 814 阅读 / 0 字 / 正在检测是否收录...
温馨提示:
本文最后更新于 2020-03-19,若内容或图片失效,请留言反馈。部分素材来自网络,若不小心影响到您的利益,请联系我们删除。

对于SpringMVC Controller单例和多例,下面举了个例子说明下.

第一次:类是多例,一个普通属性和一个静态属性。

@RestController
@Scope(value = "prototype")
public class TestController {

    private Integer num=0;
    private static Integer number=0;
    @RequestMapping("/test")
    public String test(){
        String str="普通属性:num:"+(num++)+"    ;静态属性:number:"+(number++);
        System.out.println(str);
        return str;
    }
}

结果:

普通属性:num:0    ;静态属性:number:0
普通属性:num:0    ;静态属性:number:1
普通属性:num:0    ;静态属性:number:2
普通属性:num:0    ;静态属性:number:3

所以说:对于多例情况普通属性是不会共用的,不会产生影响,对于静态属性会去共用这个属性。

第二次:类改为单例

@RestController
@Scope(value = "singleton")
public class TestController {

    private Integer num=0;
    private static Integer number=0;
    @RequestMapping("/test")
    public String test(){
        String str="普通属性:num:"+(num++)+"    ;静态属性:number:"+(number++);
        System.out.println(str);
        return str;
    }
}

结果:

普通属性:num:0    ;静态属性:number:0
普通属性:num:1    ;静态属性:number:1
普通属性:num:2    ;静态属性:number:2
普通属性:num:3    ;静态属性:number:3

所以说:对于单例情况普通属性和静态属性都会被共用。

第三次:类去掉@Scope注解

@RestController
public class TestController {

    private Integer num=0;
    private static Integer number=0;
    @RequestMapping("/test")
    public String test(){
        String str="普通属性:num:"+(num++)+"    ;静态属性:number:"+(number++);
        System.out.println(str);
        return str;
    }
}

结果:

普通属性:num:0    ;静态属性:number:0
普通属性:num:1    ;静态属性:number:1
普通属性:num:2    ;静态属性:number:2
普通属性:num:3    ;静态属性:number:3

所以说:springmvc默认是单例的。

另外在其他方法里面打印,输出的结果是一样的,跳到别的方法里面也并不会去取初始值,而是再去共用这个属性。

总结

尽量不要在controller里面去定义属性,如果在特殊情况需要定义属性的时候,那么就在类上面加上注解@Scope("prototype")改为多例的模式.

以前struts是基于类的属性进行发的,定义属性可以整个类通用,所以默认是多例,不然多线程访问肯定是共用类里面的属性值的,肯定是不安全的,但是springmvc是基于方法的开发,都是用形参接收值,一个方法结束参数就销毁了,多线程访问都会有一块内存空间产生,里面的参数也是不会共用的,所有springmvc默认使用了单例.

所以controller里面不适合在类里面定义属性,只要controller中不定义属性,那么单例完全是安全的。springmvc这样设计主要的原因也是为了提高程序的性能和以后程序的维护只针对业务的维护就行,要是struts的属性定义多了,都不知道哪个方法用了这个属性,对以后程序的维护还是很麻烦的。

0

评论区