关于spring:SpringBoot源码mvc工作流程下

42次阅读

共计 3160 个字符,预计需要花费 8 分钟才能阅读完成。

返回值解决

上一节讲到了 DispatcherServlet 依据拜访门路找到对应的类办法,在调用了类办法后,须要做返回值的解决,比方返回 json 等。关上源码,开干~

时光顺流~ 回到 doDispatch() 办法,找到这行码:

// Actually invoke the handler.
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

别怂,点进去,始终找到 RequestMappingHandlerAdapter#handleInternal() 办法,又看到一行外围代码:

// No synchronization on session demanded at all...
mav = invokeHandlerMethod(request, response, handlerMethod);

还没到,再点进去,又找到一行外围代码:

invocableMethod.invokeAndHandle(webRequest, mavContainer);

关上该办法后,下马:

public void invokeAndHandle(ServletWebRequest webRequest, ModelAndViewContainer mavContainer,Object... providedArgs) throws Exception {
    // 对申请进行解决,解析入参,返回出参
    Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs);
    ……
    try {
       // 对返回值进行解决
        this.returnValueHandlers.handleReturnValue(returnValue, getReturnValueType(returnValue), mavContainer, webRequest);
    }
    ……
}

点击 this.returnValueHandlers.handleReturnValue() 进去:

@Override
public void handleReturnValue(Object returnValue, MethodParameter,returnType, ModelAndViewContainer mavContainer, NativeWebRequest webRequest){
    // 查找返回值的处理器
    HandlerMethodReturnValueHandler handler = selectHandler(returnValue, returnType);
    ……
   // 解决了返回值
   handler.handleReturnValue(returnValue, returnType, mavContainer, 
   webRequest);
}

看一眼 selectHandler() 的私人照:

能够看到,这里是通过遍历所有的 handler,找到与 returnType 匹配的 handler,咱们看一下 handler.supportsReturnType(returnType) 是怎么解决的,比方遍历到 RequestResponseBodyMethodProcessor,关上它的 supportsReturnType() 办法:

@Override
public boolean supportsReturnType(MethodParameter returnType) {return (AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(),     
  ResponseBody.class) || returnType.hasMethodAnnotation(ResponseBody.class));
}

这里也是很简略,不就是判断有没有 @ResponseBody 这个注解咩,有的话就返回这个处理器。

回到 handleReturnValue() 办法,看那一行代码:

handler.handleReturnValue(returnValue, returnType, mavContainer, webRequest);

进入,下马

@Override
public void handleReturnValue(@Nullable Object returnValue, MethodParameter returnType,    ModelAndViewContainer mavContainer, NativeWebRequest webRequest)
throws IOException, HttpMediaTypeNotAcceptableException, HttpMessageNotWritableException {mavContainer.setRequestHandled(true);
    // 构建输入输出流
    ServletServerHttpRequest inputMessage = createInputMessage(webRequest);
    ServletServerHttpResponse outputMessage = createOutputMessage(webRequest);

    // 转换信息并写出
    writeWithMessageConverters(returnValue, returnType, inputMessage, outputMessage);
    }

点击 writeWithMessageConverters() 进入,代码很多,上有用的:

for (HttpMessageConverter<?> converter : this.messageConverters) {GenericHttpMessageConverter genericConverter = (converter instanceof GenericHttpMessageConverter ?(GenericHttpMessageConverter<?>) converter : null);
    // canWrite() 判断获取 genericConverter
    if (genericConverter != null ?
        ((GenericHttpMessageConverter) converter).canWrite(targetType, valueType, selectedMediaType) :converter.canWrite(valueType, electedMediaType))         {body = getAdvice().beforeBodyWrite(body, returnType, selectedMediaType,
            (Class<? extends HttpMessageConverter<?>>) converter.getClass(),
            inputMessage, outputMessage);
        ……
        // 返回值转换
        if (genericConverter != null) {
            genericConverter.write(body, targetType, selectedMediaType, 
                                   outputMessage);
        }
    }
}

最初找到的是 AbstractJackson2HttpMessageConverter#writeInternal(),该办法内的:

objectWriter.writeValue(generator, value);

这行代码,会调用 jackson 工具把输入值进行 json 序列化了。

再到前面是 model 和 view 的解决,都什么年代了,大部分我的项目都是前后端拆散了,这个就跳过了。

springmvc 讲完,开瓶茅台庆贺一下~~

正文完
 0