68.Spring MVC的控制器是不是单例模式,如果是,有什么问题,怎么解决?
作者:mmseoamin日期:2024-02-04

Spring MVC的控制器是不是单例模式,如果是,有什么问题,怎么解决?

在 Spring MVC 中,默认情况下控制器是单例模式,也就是说,每个控制器类只有一个实例,该实例在应用程序启动时创建,并在整个应用程序的生命周期内重复使用。这是因为单例模式可以提高应用程序的性能,减少资源消耗。

然而,如果控制器类中存在共享状态,例如实例变量,那么在多线程环境下可能会出现线程安全问题。因为多个请求处理可能会同时访问同一个控制器实例中的共享变量,导致并发问题。

为了解决这个问题,可以使用线程安全的方案来保证控制器实例的状态安全。具体做法可以是:

  • 避免使用控制器中的实例变量,可以将状态保存在请求参数中或者会话中。
  • 控制器类实现 org.springframework.web.context.request.NativeWebRequest 接口,该接口提供了一个 setAttribute() 方法,可以将状态保存在请求作用域中,而不是使用实例变量。
  • 控制器类使用 Spring 的 @Scope 注解将其作用域设置为“prototype”,这样每个请求将创建一个新的实例,从而避免多个请求共享同一个实例。

    例如,使用 @Scope(“prototype”) 注解可以将控制器设置为原型作用域,从而为每个请求创建一个新的实例:

    @Controller
    @Scope("prototype")
    public class MyController {
        // ...
    }
    

    这样,每次请求将创建一个新的控制器实例,避免多个请求共享同一个实例的问题。但这也可能会影响应用程序的性能,因为每次请求都会创建一个新的控制器实例,可能会增加资源消耗。

    最后总结

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

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

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