zoukankan      html  css  js  c++  java
  • spring mvc 的Controller类默认Scope是单例(singleton)的

    使用Spring MVC有一段时间了,之前一直使用Struts2,在struts2中action都是原型(prototype)的, 说是因为线程安全问题,对于Spring MVC中bean默认都是(singleton)单例的,那么用@Controller注解标签注入的Controller类是单例实现的?

    测试结果发现spring3中的controller默认是单例的,若是某个controller中有一个私有的变量i,所有请求到同一个controller时,使用的i变量是共用的,即若是某个请求中修改了这个变量a,则,在别的请求中能够读到这个修改的内容。 若是在@Controller之前增加@Scope("prototype"),就可以改变单例模式为多例模式

    以下是测试步骤,代码与结果.

    1. 如果是单例类型类的,那么在Controller类中的类变量应该是共享的,如果不共享,就说明Controller类不是单例。以下是测试代码:

    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
     
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.ResponseBody;
    @Controller
    public class ExampleAction {
        private int singletonInt=1;
         @RequestMapping(value = "/test")
         @ResponseBody
         public String singleton(HttpServletRequest request,
                 HttpServletResponse response) throws Exception {
             String data=request.getParameter("data");
             if(data!=null&&data.length()>0){
                 try{
                  int paramInt= Integer.parseInt(data);
                 singletonInt = singletonInt + paramInt;
                 }
                 catch(Exception ex){
                     singletonInt+=10;
                 }
             }else{
                 singletonInt+=1000;
             } 
             return String.valueOf(singletonInt);
        }
    }

    分别三次请求: http://localhost:8080/example/test.do?data=15

    得到的返回结果如下。

    第一次: singletonInt=15

    第二次: singletonInt=30

    第三次: singletonInt=45

    从以上结果可以得知,singletonInt的状态是共享的,因此Controller是单例的。

    2. 如果Controller类是单例,那么多个线程请求同一个Controller类中的同一个方法,线程是否会堵塞

    @RequestMapping(value = "/sleepdata")
    @ResponseBody
    public String switcher(HttpServletRequest request
         , HttpServletResponse response)
               throws Exception {
      String sleep = request.getParameter("sleep");
      if (sleep.equals("on")) {
          Thread.currentThread().sleep(100000);
           return "sleep on";
       } else {
           return sleep;
      }
    }  

    验证方法:分别发送两个请求,

    第一个请求:http://localhost:8080/coreplat/sleepdata.do?sleep=on

    第二个请求:http://localhost:8080/coreplat/sleepdata.do?sleep=test

    验证结果:第一个请求发出去以后,本地服务器等待100s,然后返回结果"sleep on",在本地服务器等待的者100s当中,发送第二个请求,直接返回结果"test"。说明之间的线程是不互相影响的。

  • 相关阅读:
    OOP3(继承中的类作用域/构造函数与拷贝控制/继承与容器)
    OOP2(虚函数/抽象基类/访问控制与继承)
    OOP1(定义基类和派生类)
    拷贝控制3(对象移动)
    拷贝控制2(拷贝控制和资源管理/交换操作/动态内存管理)
    拷贝控制1(拷贝、赋值与销毁)
    动态内存2(动态数组)
    python--numpy模块、spicy模块、 matplotlib模块
    Java--23种设计模式之decorator模式
    Android开发---开发文档翻译
  • 原文地址:https://www.cnblogs.com/vinplezhang/p/5885403.html
Copyright © 2011-2022 走看看