zoukankan      html  css  js  c++  java
  • springmvc在使用@ModelAttribute注解获取Request和Response会产生线程并发不安全问题

    版权声明:本文为CSDN博主「阿智CMZ」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
    原文链接:https://blog.csdn.net/weixin_37751634/article/details/93078721

    1. springmvc在获取Request和Response有很多方式:具体请看:https://www.cnblogs.com/wade-luffy/p/8867144.html
    2. 产生线程问题的代码如下:
    public class BaseController {
        protected HttpServletRequest request;
     
        protected HttpServletResponse response;
     
        protected HttpSession session;
     
        @ModelAttribute
        public void setReqAndRes(HttpServletRequest request, HttpServletResponse response) {
     
            this.request = request;
     
            this.response = response;
     
            this.session = request.getSession();
     
        }
    }

    上面的BaseController中使用ModelAttribute注解来获取request和response对象,这样基类继承该类,想要获取对象直接可以拿到,虽然这样很简便,但是这里会产生线程不安全问题,在并发量大的情况下,获取的对象可能是同一个对象,或者为null,这些都是并发造成的问题

    分析:

    1 . ModelAttribute注解有以下三个作用:
    –1)有此注解的方法会在每一个请求前执行,也就是controller执行你请求的方法之前
    –2)有此注解的参数,会从前端提交的表单中获取数据(现在一般不使用,因为不使用也可以获取到)
    –3)有此注解的方法的返回值,可以直接在spring的view中使用(也可以在BaseContoller中添加方法,传入参数 Model model,spring会自动注入model参数,给这个model添加值,也可在view中使用)

    2 . SpringMVC会优先执行被@ModelAttribute注解的方法。也就是说我们每一次请求,都会去调用init()方法,进行初始化操作,那么就会对request,response,httpSession进行赋值操作,一旦赋值,当高并发时,线程问题是必然的

     private void invokeModelAttributeMethods(NativeWebRequest request, ModelAndViewContainer container)
                throws Exception {
     
            while (!this.modelMethods.isEmpty()) {
                InvocableHandlerMethod modelMethod = getNextModelMethod(container).getHandlerMethod();
                ModelAttribute ann = modelMethod.getMethodAnnotation(ModelAttribute.class);
                if (container.containsAttribute(ann.name())) {
                    if (!ann.binding()) {
                        container.setBindingDisabled(ann.name());
                    }
                    continue;
                }
     
                Object returnValue = modelMethod.invokeForRequest(request, container);
                if (!modelMethod.isVoid()){
                    String returnValueName = getNameForReturnValue(returnValue, modelMethod.getReturnType());
                    if (!ann.binding()) {
                        container.setBindingDisabled(returnValueName);
                    }
                    if (!container.containsAttribute(returnValueName)) {
                        container.addAttribute(returnValueName, returnValue);
                    }
                }
            }
        }

    一, 可以使用其他两种方式来获取request和response对象

    1: 使用@Autowired或者@Resource注解注入,

    public abstract class BaseController {
     
        @Autowired
        protected HttpServletRequest request;
         @Autowired
         protected HttpServletResponse response;
    }

    使用@Autowired或者@Resource注解注入,是线程安全的,当用户发出请求后,会经过FrameworkServlet中的processRequest()方法做了一些骚操作,然后再交给子类DispatcherServlet中的doService()去处理这个请求。这些骚操作就包括把request,response对象包装成ServletRequestAttributes对象,然后放入到ThreadLocal中,看到ThreadLocal就明白了

        private static final ThreadLocal<RequestAttributes> requestAttributesHolder =
                new NamedThreadLocal<RequestAttributes>("Request attributes");
     
        private static final ThreadLocal<RequestAttributes> inheritableRequestAttributesHolder =
                new NamedInheritableThreadLocal<RequestAttributes>("Request context");
        private static ServletRequestAttributes currentRequestAttributes() {
            RequestAttributes requestAttr = RequestContextHolder.currentRequestAttributes();
            if (!(requestAttr instanceof ServletRequestAttributes)) {
                throw new IllegalStateException("Current request is not a servlet request");
            }
            return (ServletRequestAttributes) requestAttr;
        }

    Autowired注入流程结合上面代码:@Autowired注入HttpServletRequest对象的过程。这里以HttpServletRequest对象注入举例。首先调用DefaultListableBeanFactory中的findAutowireCandidates()方法,判断autowiringType类型是否和requiredType类型一致或者是autowiringType是否是requiredType的父接口(父类)。如果满足条件的话,我们会从resolvableDependencies中通过autowiringType(对应着上文的ServletRequest)拿到autowiringValue(对应着上文的RequestObjectFactory)。然后调用AutowireUtils.resolveAutowiringValue()对我们的ObjectFactory进行处理。

    二, 简单粗暴不会产生任何问题:直接在需要request和response对象得方法参数注入即可,有其他参数直接接着在后面写,即可,这个方法参数,作用域在方法,不会产生线程问题

     @RequestMapping(value = "/cms/preview/{id}",method = RequestMethod.GET)
     
        public void preview(HttpServletResponse response, @PathVariable("id") String id) {
     
        }

    具体详细请看:https://blog.csdn.net/weixin_33768481/article/details/88303335
    里面也介绍了进行压力测试模拟高并发请求


    ————————————————
    版权声明:本文为CSDN博主「阿智CMZ」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
    原文链接:https://blog.csdn.net/weixin_37751634/article/details/93078721

  • 相关阅读:
    关于response.getWriter()写回数据的实际发生时间点
    警惕多iframe下的同名id引起的诡异问题
    spring注入静态成员变量提示invalid setter method
    plsql+绿色版oracle连接远程数据库配置及提示缺少msvcr71.dll解决方法
    <mvc:default-servlet-handler/>导致SimpleUrlHandlerMapping失效
    maven项目下jsp文件中el表达式失效问题
    Python爬虫框架Scrapy实例(三)数据存储到MongoDB
    Python爬虫实例(五) requests+flask构建自己的电影库
    MySQL与Python交互
    Python爬虫基础(四)Requests库的使用
  • 原文地址:https://www.cnblogs.com/hanjun0612/p/11995035.html
Copyright © 2011-2022 走看看